From 36de6e54c281e165dbd694c079434230ba518f0d Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Tue, 20 Aug 2024 11:57:54 +0100 Subject: [PATCH 01/63] Start of grouping stations by manufacturer --- CumulusMX/Cumulus.cs | 65 +++++++++++++++++++++++++----------- CumulusMX/StationSettings.cs | 4 +++ 2 files changed, 50 insertions(+), 19 deletions(-) diff --git a/CumulusMX/Cumulus.cs b/CumulusMX/Cumulus.cs index c9d89413..2596e2cc 100644 --- a/CumulusMX/Cumulus.cs +++ b/CumulusMX/Cumulus.cs @@ -1346,86 +1346,68 @@ public void Initialise(int HTTPport, bool DebugEnabled, string startParms) if (StationType >= 0 && StationType < StationDesc.Length) LogConsoleMessage($"Opening station type {StationType} - {StationDesc[StationType]}"); + Manufacturer = GetStationManufacturer(StationType); switch (StationType) { case StationTypes.FineOffset: case StationTypes.FineOffsetSolar: - Manufacturer = EW; station = new FOStation(this); break; case StationTypes.VantagePro: case StationTypes.VantagePro2: - Manufacturer = DAVIS; station = new DavisStation(this); break; case StationTypes.WMR928: - Manufacturer = OREGON; station = new WMR928Station(this); break; case StationTypes.WM918: - Manufacturer = OREGON; station = new WM918Station(this); break; case StationTypes.WS2300: - Manufacturer = LACROSSE; station = new WS2300Station(this); break; case StationTypes.WMR200: - Manufacturer = OREGONUSB; station = new WMR200Station(this); break; case StationTypes.Instromet: - Manufacturer = INSTROMET; station = new ImetStation(this); break; case StationTypes.WMR100: - Manufacturer = OREGONUSB; station = new WMR100Station(this); break; case StationTypes.EasyWeather: - Manufacturer = EW; station = new EasyWeather(this); station.LoadLastHoursFromDataLogs(DateTime.Now); break; case StationTypes.WLL: - Manufacturer = DAVIS; station = new DavisWllStation(this); break; case StationTypes.GW1000: - Manufacturer = ECOWITT; station = new GW1000Station(this); break; case StationTypes.Tempest: - Manufacturer = WEATHERFLOW; station = new TempestStation(this); break; case StationTypes.HttpWund: - Manufacturer = HTTPSTATION; station = new HttpStationWund(this); break; case StationTypes.HttpEcowitt: - Manufacturer = ECOWITT; station = new HttpStationEcowitt(this); break; case StationTypes.HttpAmbient: - Manufacturer = AMBIENT; station = new HttpStationAmbient(this); break; case StationTypes.Simulator: - Manufacturer = SIMULATOR; station = new Simulator(this); break; case StationTypes.EcowittCloud: - Manufacturer = ECOWITT; station = new EcowittCloudStation(this); break; case StationTypes.DavisCloudWll: case StationTypes.DavisCloudVP2: - Manufacturer = DAVIS; station = new DavisCloudStation(this); break; case StationTypes.JsonStation: - Manufacturer = JSONSTATION; station = new JsonStation(this); break; @@ -3600,6 +3582,7 @@ private void ReadIniFile() StationType = ini.GetValue("Station", "Type", -1); StationModel = ini.GetValue("Station", "Model", string.Empty); + Manufacturer = GetStationManufacturer(StationType); FineOffsetStation = (StationType == StationTypes.FineOffset || StationType == StationTypes.FineOffsetSolar); DavisStation = (StationType == StationTypes.VantagePro || StationType == StationTypes.VantagePro2); @@ -13370,6 +13353,50 @@ public void SetupFtpLogging(bool enable) FtpLoggerMX = loggerFactory.CreateLogger("CMX"); } + private int GetStationManufacturer(int type) + { + switch (type) + { + case StationTypes.FineOffset: + case StationTypes.FineOffsetSolar: + case StationTypes.EasyWeather: + return EW; + case StationTypes.VantagePro: + case StationTypes.VantagePro2: + case StationTypes.WLL: + case StationTypes.DavisCloudWll: + case StationTypes.DavisCloudVP2: + return DAVIS; + case StationTypes.WMR928: + case StationTypes.WM918: + return OREGON; + case StationTypes.WMR200: + case StationTypes.WMR100: + return OREGONUSB; + case StationTypes.WS2300: + return LACROSSE; + case StationTypes.Instromet: + return INSTROMET; + case StationTypes.GW1000: + case StationTypes.HttpEcowitt: + case StationTypes.EcowittCloud: + return ECOWITT; + case StationTypes.Tempest: + return WEATHERFLOW; + case StationTypes.HttpWund: + return HTTPSTATION; + case StationTypes.HttpAmbient: + return AMBIENT; + case StationTypes.Simulator: + return SIMULATOR; + case StationTypes.JsonStation: + return JSONSTATION; + default: + return -1; + } + } + + [GeneratedRegex(@"max[\s]*=[\s]*([\d]+)")] private static partial Regex regexMaxParam(); [GeneratedRegex(@"[\\/]+\d{8}-\d{6}\.txt")] diff --git a/CumulusMX/StationSettings.cs b/CumulusMX/StationSettings.cs index 0b9eadcd..780a516d 100644 --- a/CumulusMX/StationSettings.cs +++ b/CumulusMX/StationSettings.cs @@ -409,6 +409,7 @@ internal string GetAlpacaFormData() var general = new JsonStationGeneral() { + manufacturer = cumulus.Manufacturer, stationtype = cumulus.StationType, stationmodel = cumulus.StationModel, loginterval = cumulus.DataLogInterval, @@ -1326,11 +1327,13 @@ internal string UpdateConfig(IHttpContext context) // Station type try { + if (cumulus.StationType != settings.general.stationtype) { cumulus.LogWarningMessage("Station type changed, restart required"); Cumulus.LogConsoleMessage("*** Station type changed, restart required ***", ConsoleColor.Yellow, true); } + cumulus.Manufacturer = settings.general.manufacturer; cumulus.StationType = settings.general.stationtype; cumulus.StationModel = settings.general.stationmodel; } @@ -1627,6 +1630,7 @@ internal class JsonStationSettingsData internal class JsonStationGeneral { + public int manufacturer { get; set; } public int stationtype { get; set; } public string stationmodel { get; set; } public int loginterval { get; set; } From 5c4a13d869f103de20b5eb375bc29c4346e7c227 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Wed, 21 Aug 2024 11:22:24 +0100 Subject: [PATCH 02/63] Add UseDataLogger to station options/advanced --- CumulusMX/Cumulus.cs | 7 +++---- CumulusMX/CumulusMX.csproj | 2 +- CumulusMX/DavisCloudStation.cs | 8 ++++++-- CumulusMX/DavisStation.cs | 2 +- CumulusMX/StationSettings.cs | 6 ++++-- Updates.txt | 13 +++++++++++++ 6 files changed, 28 insertions(+), 10 deletions(-) diff --git a/CumulusMX/Cumulus.cs b/CumulusMX/Cumulus.cs index 2596e2cc..1d748401 100644 --- a/CumulusMX/Cumulus.cs +++ b/CumulusMX/Cumulus.cs @@ -3832,7 +3832,7 @@ private void ReadIniFile() rewriteRequired = true; } - UseDataLogger = ini.GetValue("Station", "UseDataLogger", true); + StationOptions.UseDataLogger = ini.GetValue("Station", "UseDataLogger", true); UseCumulusForecast = ini.GetValue("Station", "UseCumulusForecast", false); HourlyForecast = ini.GetValue("Station", "HourlyForecast", false); StationOptions.UseCumulusPresstrendstr = ini.GetValue("Station", "UseCumulusPresstrendstr", false); @@ -5525,7 +5525,7 @@ internal void WriteIniFile() ini.SetValue("Station", "StartDateIso", RecordsBeganDateTime.ToString("yyyy-MM-dd", CultureInfo.InvariantCulture)); ini.SetValue("Station", "YTDrain", YTDrain); ini.SetValue("Station", "YTDrainyear", YTDrainyear); - ini.SetValue("Station", "UseDataLogger", UseDataLogger); + ini.SetValue("Station", "UseDataLogger", StationOptions.UseDataLogger); ini.SetValue("Station", "UseCumulusForecast", UseCumulusForecast); ini.SetValue("Station", "HourlyForecast", HourlyForecast); ini.SetValue("Station", "UseCumulusPresstrendstr", StationOptions.UseCumulusPresstrendstr); @@ -7201,8 +7201,6 @@ public void WriteStringsFile() public bool UseCumulusForecast { get; set; } - public bool UseDataLogger { get; set; } - public bool DavisConsoleHighGust { get; set; } public bool DavisCalcAltPress { get; set; } @@ -13535,6 +13533,7 @@ public class StationOptions public int UseRainForIsRaining { get; set; } public int LeafWetnessIsRainingIdx { get; set; } public double LeafWetnessIsRainingThrsh { get; set; } + public bool UseDataLogger { get; set; } } public class FtpOptionsClass diff --git a/CumulusMX/CumulusMX.csproj b/CumulusMX/CumulusMX.csproj index d58ab862..d2a1d572 100644 --- a/CumulusMX/CumulusMX.csproj +++ b/CumulusMX/CumulusMX.csproj @@ -55,7 +55,7 @@ - 4.1.3.4028 + 4.2.0.4029 Copyright © 2015-$([System.DateTime]::Now.ToString('yyyy')) Cumulus MX diff --git a/CumulusMX/DavisCloudStation.cs b/CumulusMX/DavisCloudStation.cs index c780c113..0c01a09a 100644 --- a/CumulusMX/DavisCloudStation.cs +++ b/CumulusMX/DavisCloudStation.cs @@ -139,7 +139,7 @@ public DavisCloudStation(Cumulus cumulus) : base(cumulus) DateTime tooOld = new DateTime(0, DateTimeKind.Local); - if ((cumulus.LastUpdateTime <= tooOld) || !cumulus.UseDataLogger) + if ((cumulus.LastUpdateTime <= tooOld) || !cumulus.StationOptions.UseDataLogger) { // there's nothing in the database, so we haven't got a rain counter // we can't load the history data, so we'll just have to go live @@ -494,6 +494,7 @@ private void GetHistoricData(BackgroundWorker worker) cumulus.LogErrorMessage($"GetHistoricData: WeatherLink API Historic Error: {historyError.code}, {historyError.message}"); Cumulus.LogConsoleMessage($" - Error {historyError.code}: {historyError.message}", ConsoleColor.Red); //cumulus.LastUpdateTime = Utils.FromUnixTime(endTime) + maxArchiveRuns = 0; return; } @@ -502,6 +503,7 @@ private void GetHistoricData(BackgroundWorker worker) cumulus.LogWarningMessage("GetHistoricData: WeatherLink API Historic: No data was returned. Check your Device Id."); Cumulus.LogConsoleMessage(" - No historic data available"); lastHistoricData = Utils.FromUnixTime(endTime); + maxArchiveRuns = 0; return; } else if (responseBody.StartsWith("{\"")) // basic sanity check @@ -542,6 +544,7 @@ private void GetHistoricData(BackgroundWorker worker) cumulus.LogErrorMessage("GetHistoricData: Invalid historic message received"); cumulus.LogMessage("GetHistoricData: Received: " + responseBody); lastHistoricData = Utils.FromUnixTime(endTime); + maxArchiveRuns = 0; return; } } @@ -555,6 +558,7 @@ private void GetHistoricData(BackgroundWorker worker) } lastHistoricData = Utils.FromUnixTime(endTime); + maxArchiveRuns = 0; return; } @@ -2104,7 +2108,7 @@ private void DecodeCurrent(List sensors) // If the station isn't using the logger function for WLL - i.e. no API key, then only alarm on Tx battery status // otherwise, trigger the alarm when we read the Health data which also contains the WLL backup battery status - if (!cumulus.UseDataLogger) + if (!cumulus.StationOptions.UseDataLogger) { cumulus.BatteryLowAlarm.Triggered = TxBatText.Contains("LOW"); } diff --git a/CumulusMX/DavisStation.cs b/CumulusMX/DavisStation.cs index 338f909c..5447c751 100644 --- a/CumulusMX/DavisStation.cs +++ b/CumulusMX/DavisStation.cs @@ -195,7 +195,7 @@ public DavisStation(Cumulus cumulus) : base(cumulus) DateTime tooold = new DateTime(0, DateTimeKind.Local); - if ((cumulus.LastUpdateTime <= tooold) || !cumulus.UseDataLogger) + if ((cumulus.LastUpdateTime <= tooold) || !cumulus.StationOptions.UseDataLogger) { // there's nothing in the database, so we haven't got a rain counter // we can't load the history data, so we'll just have to go live diff --git a/CumulusMX/StationSettings.cs b/CumulusMX/StationSettings.cs index 780a516d..31426315 100644 --- a/CumulusMX/StationSettings.cs +++ b/CumulusMX/StationSettings.cs @@ -48,7 +48,8 @@ internal string GetAlpacaFormData() maxwind = cumulus.LCMaxWind, recordtimeout = cumulus.RecordSetTimeoutHrs, snowdepthhour = cumulus.SnowDepthHour, - raindaythreshold = cumulus.RainDayThreshold + raindaythreshold = cumulus.RainDayThreshold, + uselogger = cumulus.StationOptions.UseDataLogger }; // Common Settings @@ -710,11 +711,11 @@ internal string UpdateConfig(IHttpContext context) cumulus.StationOptions.AvgBearingMinutes = settings.Options.advanced.avgbearingmins; cumulus.StationOptions.AvgSpeedMinutes = settings.Options.advanced.avgspeedmins; cumulus.StationOptions.PeakGustMinutes = settings.Options.advanced.peakgustmins; + cumulus.StationOptions.UseDataLogger = settings.Options.advanced.uselogger; cumulus.LCMaxWind = settings.Options.advanced.maxwind; cumulus.RecordSetTimeoutHrs = settings.Options.advanced.recordtimeout; cumulus.SnowDepthHour = settings.Options.advanced.snowdepthhour; cumulus.RainDayThreshold = settings.Options.advanced.raindaythreshold; - } catch (Exception ex) { @@ -1680,6 +1681,7 @@ internal class JsonStationSettingsOptionsAdvanced public int recordtimeout { get; set; } public int snowdepthhour { get; set; } public double raindaythreshold { get; set; } + public bool uselogger { get; set; } } internal class JsonStationSettingsOptions diff --git a/Updates.txt b/Updates.txt index d4108f2e..f1296760 100644 --- a/Updates.txt +++ b/Updates.txt @@ -1,3 +1,16 @@ +4.2.0 - b4029 +————————————— +New +- Exposes the UseDataLogger setting in Station Settings > Options > Advanced + +Changed + + +Fixed +- Davis Cloud stations no longer continuosly try to fetch history data if there is no Pro subscription + + + 4.1.3 - b4028 ————————————— From c5c2c2140ed4db85108114026db20d6d98f68d99 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Wed, 21 Aug 2024 11:26:49 +0100 Subject: [PATCH 03/63] Davis cloud station now uses subscription level to determine whether to fecth historic data --- CumulusMX/DavisCloudStation.cs | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/CumulusMX/DavisCloudStation.cs b/CumulusMX/DavisCloudStation.cs index 0c01a09a..9a87d3b5 100644 --- a/CumulusMX/DavisCloudStation.cs +++ b/CumulusMX/DavisCloudStation.cs @@ -25,6 +25,7 @@ internal partial class DavisCloudStation : WeatherStation private bool startingUp = true; private DateTime lastRecordTime = DateTime.MinValue; private DateTime lastHistoricData; + private string subscriptionLevel = "basic"; public DavisCloudStation(Cumulus cumulus) : base(cumulus) { @@ -139,7 +140,7 @@ public DavisCloudStation(Cumulus cumulus) : base(cumulus) DateTime tooOld = new DateTime(0, DateTimeKind.Local); - if ((cumulus.LastUpdateTime <= tooOld) || !cumulus.StationOptions.UseDataLogger) + if ((cumulus.LastUpdateTime <= tooOld) || subscriptionLevel == "basic" || !cumulus.StationOptions.UseDataLogger) { // there's nothing in the database, so we haven't got a rain counter // we can't load the history data, so we'll just have to go live @@ -3753,9 +3754,9 @@ private void GetAvailableStationIds(bool logToConsole = false) private void SetDataTimeout(string subscription) { - subscription = (subscription ?? "basic").ToLower(); + subscriptionLevel = (subscription ?? "basic").ToLower(); - DataTimeoutMins = subscription switch + DataTimeoutMins = subscriptionLevel switch { "basic" => 15 + 3, "pro" => 5 + 3, From f9b8c0a8c7a4cef416a7c8d08bc49933f2e70b19 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Wed, 21 Aug 2024 11:32:29 +0100 Subject: [PATCH 04/63] Mop up some more UseDataLogger references --- CumulusMX/DavisWllStation.cs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/CumulusMX/DavisWllStation.cs b/CumulusMX/DavisWllStation.cs index b618bc8a..d436112c 100644 --- a/CumulusMX/DavisWllStation.cs +++ b/CumulusMX/DavisWllStation.cs @@ -190,7 +190,7 @@ public DavisWllStation(Cumulus cumulus) : base(cumulus) DateTime tooOld = new DateTime(0, DateTimeKind.Local); - if ((cumulus.LastUpdateTime <= tooOld) || !cumulus.UseDataLogger) + if ((cumulus.LastUpdateTime <= tooOld) || !cumulus.StationOptions.UseDataLogger) { // there's nothing in the database, so we haven't got a rain counter // we can't load the history data, so we'll just have to go live @@ -1312,7 +1312,7 @@ private void DecodeCurrent(string currentJson) // If the station isn't using the logger function for WLL - i.e. no API key, then only alarm on Tx battery status // otherwise, trigger the alarm when we read the Health data which also contains the WLL backup battery status - if (!cumulus.UseDataLogger) + if (!cumulus.StationOptions.UseDataLogger) { cumulus.BatteryLowAlarm.Triggered = TxBatText.Contains("LOW"); } From 0dc5e33edc2964c3c2c654aee6472bb81c2232e7 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Wed, 21 Aug 2024 11:53:23 +0100 Subject: [PATCH 05/63] More subcription based decisions in Davis WLL+Cloud stations --- CumulusMX/DavisCloudStation.cs | 5 ++--- CumulusMX/DavisWllStation.cs | 9 +++++++-- 2 files changed, 9 insertions(+), 5 deletions(-) diff --git a/CumulusMX/DavisCloudStation.cs b/CumulusMX/DavisCloudStation.cs index 9a87d3b5..4add8540 100644 --- a/CumulusMX/DavisCloudStation.cs +++ b/CumulusMX/DavisCloudStation.cs @@ -3693,10 +3693,10 @@ private void GetAvailableStationIds(bool logToConsole = false) foreach (var station in stationsObj.stations) { - cumulus.LogMessage($"GetStations: Found WeatherLink station id = {station.station_id}, name = {station.station_name}"); + cumulus.LogMessage($"GetStations: WeatherLink station id = {station.station_id}, name = {station.station_name}, subscription = {station.subscription_type}"); if (stationsObj.stations.Count > 1 && logToConsole) { - Cumulus.LogConsoleMessage($" - Found WeatherLink station id = {station.station_id}, name = {station.station_name}, active = {station.active}"); + Cumulus.LogConsoleMessage($" - Found WeatherLink station id = {station.station_id}, name = {station.station_name}, active = {station.active}, subscription = {station.subscription_type}"); } if (station.station_id == cumulus.WllStationId || station.station_id_uuid == cumulus.WllStationUuid) { @@ -3711,7 +3711,6 @@ private void GetAvailableStationIds(bool logToConsole = false) wlStationArchiveInterval = station.recording_interval; SetDataTimeout(station.subscription_type); - if (cumulus.WllStationId < 10) { cumulus.WllStationId = station.station_id; diff --git a/CumulusMX/DavisWllStation.cs b/CumulusMX/DavisWllStation.cs index d436112c..7f7c0cbd 100644 --- a/CumulusMX/DavisWllStation.cs +++ b/CumulusMX/DavisWllStation.cs @@ -44,6 +44,7 @@ internal partial class DavisWllStation : WeatherStation private readonly bool useWeatherLinkDotCom = true; private readonly bool[] sensorContactLost = new bool[9]; private DateTime lastHistoricData; + private string subscriptionType = string.Empty; public DavisWllStation(Cumulus cumulus) : base(cumulus) { @@ -3039,10 +3040,10 @@ private void GetAvailableStationIds(bool logToConsole = false) foreach (var station in stationsObj.stations) { - cumulus.LogMessage($"WLLStations: Found WeatherLink station id = {station.station_id}, name = {station.station_name}"); + cumulus.LogMessage($"WLLStations: WeatherLink station id = {station.station_id}, name = {station.station_name}, active = {station.active}, subscription = {station.subscription_type}"); if (stationsObj.stations.Count > 1 && logToConsole) { - Cumulus.LogConsoleMessage($" - Found WeatherLink station id = {station.station_id}, name = {station.station_name}, active = {station.active}"); + Cumulus.LogConsoleMessage($" - Found WeatherLink station id = {station.station_id}, name = {station.station_name}, active = {station.active}, subscription = {station.subscription_type}"); } if (station.station_id == cumulus.WllStationId || cumulus.WllStationUuid == station.station_id_uuid) { @@ -3063,6 +3064,8 @@ private void GetAvailableStationIds(bool logToConsole = false) cumulus.WllStationUuid = station.station_id_uuid; } + subscriptionType = station.subscription_type.ToLower(); + cumulus.WriteIniFile(); } } @@ -3078,6 +3081,8 @@ private void GetAvailableStationIds(bool logToConsole = false) cumulus.LogMessage($"WLLStations: Only found 1 WeatherLink station, using id = {usedId}"); cumulus.WllStationId = stationsObj.stations[0].station_id; cumulus.WllStationUuid = stationsObj.stations[0].station_id_uuid; + subscriptionType = stationsObj.stations[0].subscription_type.ToLower(); + // And save it to the config file cumulus.WriteIniFile(); From 9864c80109439b1c81452d291bc9464493404d31 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Thu, 22 Aug 2024 09:46:27 +0100 Subject: [PATCH 06/63] Start of Ecowitt local HTTP API station --- CumulusMX/Cumulus.cs | 5 + CumulusMX/EcowittHttpApiStation.cs | 1850 ++++++++++++++++++++++++++++ CumulusMX/EcowittLocalApi.cs | 168 +++ 3 files changed, 2023 insertions(+) create mode 100644 CumulusMX/EcowittHttpApiStation.cs create mode 100644 CumulusMX/EcowittLocalApi.cs diff --git a/CumulusMX/Cumulus.cs b/CumulusMX/Cumulus.cs index 1d748401..91b7b36c 100644 --- a/CumulusMX/Cumulus.cs +++ b/CumulusMX/Cumulus.cs @@ -1410,6 +1410,9 @@ public void Initialise(int HTTPport, bool DebugEnabled, string startParms) case StationTypes.JsonStation: station = new JsonStation(this); break; + case StationTypes.EcowittHttpApi: + station = new EcowittHttpApiStation(this); + break; default: LogConsoleMessage("Station type not set", ConsoleColor.Red); @@ -13378,6 +13381,7 @@ private int GetStationManufacturer(int type) case StationTypes.GW1000: case StationTypes.HttpEcowitt: case StationTypes.EcowittCloud: + case StationTypes.EcowittHttpApi: return ECOWITT; case StationTypes.Tempest: return WEATHERFLOW; @@ -13430,6 +13434,7 @@ public static class StationTypes public const int DavisCloudWll = 19; public const int DavisCloudVP2 = 20; public const int JsonStation = 21; + public const int EcowittHttpApi = 22; } public class DiaryData diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs new file mode 100644 index 00000000..f2afbe1a --- /dev/null +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -0,0 +1,1850 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Net; +using System.Net.Sockets; +using System.Text; +using System.Threading; +using System.Threading.Tasks; +using System.Timers; + +using ServiceStack.Text; + +namespace CumulusMX +{ +#pragma warning disable CA1001 // Types that own disposable fields should be disposable + internal class EcowittHttpApiStation : WeatherStation +#pragma warning restore CA1001 // Types that own disposable fields should be disposable + { + private string ipaddr; + private readonly string macaddr; + private string deviceModel; + private string deviceFirmware; + private int updateRate = 10000; // 10 seconds by default + private int lastMinute = -1; + private int lastHour = -1; + + private readonly EcowittApi ecowittApi; + private readonly GW1000Api Api; + + private int maxArchiveRuns = 1; + + private bool dataReceived = false; + + private readonly System.Timers.Timer tmrDataWatchdog; + + private readonly Task historyTask; + private Task liveTask; + + private Version fwVersion; + internal static readonly char[] dotSeparator = ['.']; + internal static readonly string[] underscoreV = ["_V"]; + + public GW1000Station(Cumulus cumulus) : base(cumulus) + { + cumulus.Units.AirQualityUnitText = "µg/m³"; + cumulus.Units.SoilMoistureUnitText = "%"; + cumulus.Units.LeafWetnessUnitText = "%"; + + // GW1000 does not provide 10 min average wind speeds + cumulus.StationOptions.CalcuateAverageWindSpeed = true; + + // GW1000 does not provide an interval gust value, it gives us a 30 second high + // so force using the wind speed for the average calculation + cumulus.StationOptions.UseSpeedForAvgCalc = true; + // also use it for the Latest value + cumulus.StationOptions.UseSpeedForLatest = true; + + LightningTime = DateTime.MinValue; + LightningDistance = -1.0; + + tmrDataWatchdog = new System.Timers.Timer(); + + // GW1000 does not send DP, so force MX to calculate it + cumulus.StationOptions.CalculatedDP = true; + + // does not provide a forecast, force MX to provide it + cumulus.UseCumulusForecast = true; + + // GW1000 does not provide pressure trend strings + cumulus.StationOptions.UseCumulusPresstrendstr = true; + + if (cumulus.Gw1000PrimaryTHSensor == 0) + { + // We are using the primary T/H sensor + cumulus.LogMessage("Using the default outdoor temp/hum sensor data"); + } + else if (cumulus.Gw1000PrimaryTHSensor == 99) + { + // We are overriding the outdoor with the indoor T/H sensor + cumulus.LogMessage("Overriding the default outdoor temp/hum data with Indoor temp/hum sensor"); + cumulus.StationOptions.CalculatedDP = true; + cumulus.StationOptions.CalculatedWC = true; + } + else + { + // We are not using the primary T/H sensor so MX must calculate the wind chill as well + cumulus.StationOptions.CalculatedDP = true; + cumulus.StationOptions.CalculatedWC = true; + cumulus.LogMessage("Overriding the default outdoor temp/hum data with Extra temp/hum sensor #" + cumulus.Gw1000PrimaryTHSensor); + } + + if (cumulus.Gw1000PrimaryRainSensor == 0) + { + // We are using the traditional rain tipper + cumulus.LogMessage("Using the default traditional rain sensor data"); + } + else + { + cumulus.LogMessage("Using the piezo rain sensor data"); + } + + ipaddr = cumulus.Gw1000IpAddress; + macaddr = cumulus.Gw1000MacAddress; + + Api = new GW1000Api(cumulus); + + ecowittApi = new EcowittApi(cumulus, this); + + DoDiscovery(); + PostDiscovery(); + + _ = CheckAvailableFirmware(); + + LoadLastHoursFromDataLogs(cumulus.LastUpdateTime); + + historyTask = Task.Run(getAndProcessHistoryData, cumulus.cancellationToken); + } + + + public override void Start() + { + lastMinute = DateTime.Now.Minute; + + // Start a broadcast watchdog to warn if messages are not being received + tmrDataWatchdog.Elapsed += DataTimeout; + tmrDataWatchdog.Interval = 1000 * 30; // timeout after 30 seconds + tmrDataWatchdog.AutoReset = true; + tmrDataWatchdog.Start(); + + + // just incase we did not catch-up any history + DoDayResetIfNeeded(); + DoTrendValues(DateTime.Now); + + cumulus.LogMessage("Starting Ecowitt Local API"); + + cumulus.StartTimersAndSensors(); + + liveTask = Task.Run(() => + { + try + { + var piezoLastRead = DateTime.MinValue; + var dataLastRead = DateTime.MinValue; + double delay; + + while (!cumulus.cancellationToken.IsCancellationRequested) + { + if (Api.Connected) + { + GetLiveData(); + dataLastRead = DateTime.Now; + + // every 30 seconds read the rain rate + if ((cumulus.Gw1000PrimaryRainSensor == 1 || cumulus.StationOptions.UseRainForIsRaining == 2) && (DateTime.UtcNow - piezoLastRead).TotalSeconds >= 30 && !cumulus.cancellationToken.IsCancellationRequested) + { + GetPiezoRainData(); + piezoLastRead = DateTime.UtcNow; + } + + var minute = DateTime.Now.Minute; + if (minute != lastMinute) + { + lastMinute = minute; + + // at the start of every 20 minutes to trigger battery status check + if ((minute % 20) == 0 && !cumulus.cancellationToken.IsCancellationRequested) + { + GetSensorIdsNew(); + } + + // every day dump the clock drift at midday each day + if (minute == 0 && DateTime.Now.Hour == 12) + { + GetSystemInfo(true); + } + + var hour = DateTime.Now.Hour; + if (lastHour != hour) + { + lastHour = hour; + + if (hour == 13) + { + var fw = GetFirmwareVersion(); + if (fw != "???") + { + GW1000FirmwareVersion = fw; + deviceModel = GW1000FirmwareVersion.Split('_')[0]; + deviceFirmware = GW1000FirmwareVersion.Split('_')[1]; + + var fwString = GW1000FirmwareVersion.Split(underscoreV, StringSplitOptions.None); + if (fwString.Length > 1) + { + fwVersion = new Version(fwString[1]); + } + else + { + // failed to get the version, lets assume it's fairly new + fwVersion = new Version("1.6.5"); + } + } + + _ = CheckAvailableFirmware(); + } + } + } + } + else + { + cumulus.LogMessage("Attempting to reconnect to Ecowitt device..."); + Api.OpenTcpPort(cumulus.Gw1000IpAddress, AtPort); + if (Api.Connected) + { + cumulus.LogMessage("Reconnected to Ecowitt device"); + GetLiveData(); + } + else + { + // add a small extra delay before trying again + cumulus.LogMessage("Delaying before attempting reconnect"); + if (cumulus.cancellationToken.WaitHandle.WaitOne(TimeSpan.FromMilliseconds(20000))) + { + break; + } + } + } + + delay = Math.Min(updateRate - (dataLastRead - DateTime.Now).TotalMilliseconds, updateRate); + + if (cumulus.cancellationToken.WaitHandle.WaitOne(TimeSpan.FromMilliseconds(delay))) + { + break; + } + } + } + // Catch the ThreadAbortException + catch (ThreadAbortException) + { + //do nothing + } + finally + { + Api.CloseTcpPort(); + cumulus.LogMessage("Local API task ended"); + } + }, cumulus.cancellationToken); + } + + public override void Stop() + { + cumulus.LogMessage("Closing connection"); + try + { + tmrDataWatchdog.Stop(); + StopMinuteTimer(); + Task.WaitAll(historyTask, liveTask); + } + catch + { + //do nothing + } + } + + + public override void getAndProcessHistoryData() + { + Cumulus.SyncInit.Wait(); + + if (string.IsNullOrEmpty(cumulus.EcowittApplicationKey) || string.IsNullOrEmpty(cumulus.EcowittUserApiKey) || string.IsNullOrEmpty(cumulus.EcowittMacAddress)) + { + cumulus.LogWarningMessage("API.GetHistoricData: Missing Ecowitt API data in the configuration, aborting!"); + cumulus.LastUpdateTime = DateTime.Now; + } + else + { + int archiveRun = 0; + + try + { + do + { + GetHistoricData(); + archiveRun++; + } while (archiveRun < maxArchiveRuns && !cumulus.cancellationToken.IsCancellationRequested); + } + catch (Exception ex) + { + cumulus.LogErrorMessage("Exception occurred reading archive data: " + ex.Message); + } + + // get the station list + ecowittApi.GetStationList(true, cumulus.EcowittMacAddress, cumulus.cancellationToken); + } + + _ = Cumulus.SyncInit.Release(); + + if (cumulus.cancellationToken.IsCancellationRequested) + { + return; + } + + StartLoop(); + } + + private void GetHistoricData() + { + cumulus.LogMessage("GetHistoricData: Starting Historic Data Process"); + + // add one minute to the time to avoid duplicating the last log entry + var startTime = cumulus.LastUpdateTime.AddMinutes(1); + var endTime = DateTime.Now; + + // The API call is limited to fetching 24 hours of data + if ((endTime - startTime).TotalHours > 24.0) + { + // only fetch 24 hours worth of data, and schedule another run to fetch the rest + endTime = startTime.AddHours(24); + maxArchiveRuns++; + } + + ecowittApi.GetHistoricData(startTime, endTime, cumulus.cancellationToken); + } + + public override string GetEcowittCameraUrl() + { + if (!string.IsNullOrEmpty(cumulus.EcowittCameraMacAddress)) + { + try + { + EcowittCameraUrl = ecowittApi.GetCurrentCameraImageUrl(EcowittCameraUrl, cumulus.cancellationToken); + return EcowittCameraUrl; + } + catch (Exception ex) + { + cumulus.LogExceptionMessage(ex, "Error runing Ecowitt Camera URL"); + } + } + + return null; + } + + public override string GetEcowittVideoUrl() + { + if (!string.IsNullOrEmpty(cumulus.EcowittCameraMacAddress)) + { + try + { + EcowittVideoUrl = ecowittApi.GetLastCameraVideoUrl(EcowittVideoUrl, cumulus.cancellationToken); + return EcowittVideoUrl; + } + catch (Exception ex) + { + cumulus.LogExceptionMessage(ex, "Error running Ecowitt Video URL"); + } + } + + return null; + } + + + private Discovery DiscoverGW1000() + { + // We only want unique IP addresses + var discovered = new Discovery(); + const int broadcastPort = 46000; + + try + { + using var client = new UdpClient(); + var recvEp = new IPEndPoint(0, 0); + var sendEp = new IPEndPoint(IPAddress.Broadcast, broadcastPort); + var sendBytes = new byte[] { 0xff, 0xff, 0x12, 0x00, 0x04, 0x16 }; + + + // Get the primary IP address + var myIP = Utils.GetIpWithDefaultGateway(); + cumulus.LogDebugMessage($"Using local IP address {myIP} to discover the Ecowitt device"); + + // bind the cient to the primary address - broadcast does not work with .Any address :( + client.Client.Bind(new IPEndPoint(myIP, broadcastPort)); + // time out listening after 1.5 second + client.Client.ReceiveTimeout = 1500; + + // we are going to attempt discovery twice + var retryCount = 1; + do + { + cumulus.LogDebugMessage("Discovery Run #" + retryCount); + // each time we wait 1.5 second for any responses + var endTime = DateTime.Now.AddSeconds(1.5); + + try + { + // Send the request + client.Send(sendBytes, sendBytes.Length, sendEp); + + do + { + try + { + // get a response + var recevBuffer = client.Receive(ref recvEp); + + // sanity check the response size - we may see our request back as a receive packet + if (recevBuffer.Length > 20) + { + string ipAddr = $"{recevBuffer[11]}.{recevBuffer[12]}.{recevBuffer[13]}.{recevBuffer[14]}"; + var macArr = new byte[6]; + + Array.Copy(recevBuffer, 5, macArr, 0, 6); + var macHex = BitConverter.ToString(macArr).Replace('-', ':'); + + var nameLen = recevBuffer[17]; + var nameArr = new byte[nameLen]; + Array.Copy(recevBuffer, 18, nameArr, 0, nameLen); + var name = Encoding.UTF8.GetString(nameArr, 0, nameArr.Length); + + if (ipAddr.Split(dotSeparator, StringSplitOptions.RemoveEmptyEntries).Length == 4) + { + IPAddress ipAddr2; + if (IPAddress.TryParse(ipAddr, out ipAddr2)) + { + cumulus.LogDebugMessage($"Discovered Ecowitt device: {name}, IP={ipAddr}, MAC={macHex}"); + if (!discovered.IP.Contains(ipAddr) && !discovered.Mac.Contains(macHex)) + { + discovered.Name.Add(name); + discovered.IP.Add(ipAddr); + discovered.Mac.Add(macHex); + } + } + } + else + { + cumulus.LogDebugMessage($"Discovered an unsupported device: {name}, IP={ipAddr}, MAC={macHex}"); + } + } + } + catch + { + //do nothing + } + + } while (DateTime.Now < endTime); + } + catch (Exception ex) + { + cumulus.LogMessage("DiscoverGW1000: Error sending discovery request"); + cumulus.LogErrorMessage("DiscoverGW1000: Error: " + ex.Message); + } + retryCount++; + + } while (retryCount <= 2); + } + catch (Exception ex) + { + cumulus.LogErrorMessage("An error occurred during Ecowitt auto-discovery"); + cumulus.LogMessage("Error: " + ex.Message); + } + + return discovered; + } + + private void DoDiscovery() + { + if (cumulus.Gw1000AutoUpdateIpAddress || string.IsNullOrWhiteSpace(cumulus.Gw1000IpAddress)) + { + string msg; + cumulus.LogMessage("Running Ecowitt Local API auto-discovery..."); + cumulus.LogMessage($"Current IP address={cumulus.Gw1000IpAddress}, current MAC={cumulus.Gw1000MacAddress}"); + + var discoveredDevices = DiscoverGW1000(); + + if (discoveredDevices.IP.Count == 0) + { + // We didn't find anything on the network + msg = "Failed to discover any Ecowitt devices"; + cumulus.LogWarningMessage(msg); + Cumulus.LogConsoleMessage(msg, ConsoleColor.DarkYellow, true); + return; + } + else if (discoveredDevices.IP.Count == 1 && (string.IsNullOrEmpty(macaddr) || discoveredDevices.Mac[0] == macaddr)) + { + cumulus.LogDebugMessage("Discovered one Ecowitt device"); + // If only one device is discovered, and its MAC address matches (or our MAC is blank), then just use it + if (cumulus.Gw1000IpAddress != discoveredDevices.IP[0]) + { + cumulus.LogWarningMessage("Discovered a new IP address for the Ecowitt device that does not match our current one"); + cumulus.LogMessage($"Changing previous IP address: {ipaddr} to {discoveredDevices.IP[0]}"); + ipaddr = discoveredDevices.IP[0].Trim(); + cumulus.Gw1000IpAddress = ipaddr; + deviceModel = discoveredDevices.Name[0].Split('-')[0]; + deviceFirmware = discoveredDevices.Name[0].Split('-')[1].Split(' ')[1]; + if (discoveredDevices.Mac[0] != macaddr) + { + cumulus.Gw1000MacAddress = discoveredDevices.Mac[0].ToUpper(); + } + cumulus.WriteIniFile(); + } + else + { + cumulus.LogMessage("The discovered IP address for the GW1000 matches our current one"); + } + } + else if (discoveredDevices.Mac.Contains(macaddr)) + { + // Multiple devices discovered, but we have a MAC address match + + cumulus.LogDebugMessage("Matching Ecowitt MAC address found on the network"); + + var idx = discoveredDevices.Mac.IndexOf(macaddr); + deviceModel = discoveredDevices.Name[idx].Split('-')[0]; + deviceFirmware = discoveredDevices.Name[idx].Split('-')[1].Split(' ')[1]; + + if (discoveredDevices.IP[idx] != ipaddr) + { + cumulus.LogMessage("Discovered a new IP address for the Ecowitt device that does not match our current one"); + cumulus.LogMessage($"Changing previous IP address: {ipaddr} to {discoveredDevices.IP[idx]}"); + ipaddr = discoveredDevices.IP[idx]; + cumulus.Gw1000IpAddress = ipaddr; + cumulus.WriteIniFile(); + } + } + else + { + // Multiple devices discovered, and we do not have a clue! + + StringBuilder iplist = new(" discovered IPs = "); + msg = "Discovered more than one potential Ecowitt device."; + cumulus.LogWarningMessage(msg); + Cumulus.LogConsoleMessage(msg); + msg = "Please select the IP address from the list and enter it manually into the configuration"; + cumulus.LogMessage(msg); + Cumulus.LogConsoleMessage(msg); + + for (var i = 0; i < discoveredDevices.IP.Count; i++) + { + msg = $"Device={discoveredDevices.Name[i]}, IP={discoveredDevices.IP[i]}"; + Cumulus.LogConsoleMessage(msg); + cumulus.LogMessage(msg); + iplist.Append(discoveredDevices.IP[i] + " "); + } + cumulus.LogMessage(iplist.ToString()); + return; + } + } + + if (string.IsNullOrWhiteSpace(ipaddr)) + { + var msg = "No IP address configured or discovered for your GW1000, please remedy and restart Cumulus MX"; + cumulus.LogErrorMessage(msg); + Cumulus.LogConsoleMessage(msg); + return; + } + + return; + } + + private void PostDiscovery() + { + cumulus.LogMessage("Using IP address = " + ipaddr + " Port = " + AtPort); + + Api.OpenTcpPort(ipaddr, AtPort); + + if (Api.Connected) + { + cumulus.LogMessage("Connected OK"); + Cumulus.LogConsoleMessage("Connected to station", ConsoleColor.White, true); + } + else + { + cumulus.LogErrorMessage("Not Connected"); + Cumulus.LogConsoleMessage("Unable to connect to station", ConsoleColor.Red, true); + } + + if (Api.Connected) + { + // Get the firmware version as check we are communicating + GW1000FirmwareVersion = GetFirmwareVersion(); + cumulus.LogMessage($"Ecowitt firmware version: {GW1000FirmwareVersion}"); + if (GW1000FirmwareVersion != "???") + { + deviceModel = GW1000FirmwareVersion.Split('_')[0]; + deviceFirmware = GW1000FirmwareVersion.Split('_')[1]; + + var fwString = GW1000FirmwareVersion.Split(underscoreV, StringSplitOptions.None); + if (fwString.Length > 1) + { + fwVersion = new Version(fwString[1]); + } + else + { + // failed to get the version, lets assume it's fairly new + fwVersion = new Version("1.6.5"); + } + } + + GetSystemInfo(false); + + GetSensorIdsNew(); + } + } + + private string GetFirmwareVersion() + { + var response = "???"; + cumulus.LogMessage("Reading firmware version"); + try + { + var data = Api.DoCommand(GW1000Api.Commands.CMD_READ_FIRMWARE_VERSION); + if (null != data && data.Length > 0) + { + response = Encoding.ASCII.GetString(data, 5, data[4]); + } + } + catch (Exception ex) + { + cumulus.LogErrorMessage($"GetFirmwareVersion: Error retrieving/processing firmware version. Message - {ex.Message}"); + } + return response; + } + + private async Task CheckAvailableFirmware() + { + if (deviceModel == null) + { + cumulus.LogMessage("Device Model not determined, firmware check skipped."); + return; + } + + if (EcowittApi.FirmwareSupportedModels.Contains(deviceModel[..6])) + { + _ = await ecowittApi.GetLatestFirmwareVersion(deviceModel, cumulus.EcowittMacAddress, deviceFirmware, cumulus.cancellationToken); + } + else + { + var retVal = ecowittApi.GetSimpleLatestFirmwareVersion(deviceModel, cumulus.cancellationToken).Result; + if (retVal != null) + { + var verVer = new Version(retVal[0]); + if (fwVersion < verVer) + { + cumulus.FirmwareAlarm.LastMessage = $"A new firmware version is available: {retVal[0]}.\nChange log:\n{string.Join('\n', retVal[1].Split(';'))}"; + cumulus.FirmwareAlarm.Triggered = true; + cumulus.LogWarningMessage($"FirmwareVersion: Latest Version {retVal[0]}, Change log:\n{string.Join('\n', retVal[1].Split(';'))}"); + } + else + { + cumulus.FirmwareAlarm.Triggered = false; + cumulus.LogDebugMessage($"FirmwareVersion: Already on the latest Version {retVal[0]}"); + } + } + } + } + + private void GetSensorIdsNew() + { + cumulus.LogMessage("Reading sensor ids"); + + var data = Api.DoCommand(GW1000Api.Commands.CMD_READ_SENSOR_ID_NEW); + + // expected response + // 0 - 0xff - header + // 1 - 0xff - header + // 2 - 0x3C - sensor id command + // 3-4 - 0x?? - size of response + // 5 - wh65 + // 6-9 - wh65 id + // 10 - wh65 battery + // 11 - wh65 signal 0-4 + // 12 - wh68 + // ... etc + // (??) - 0x?? - checksum + + var batteryLow = false; + + try + { + if (null != data && data.Length > 200) + { + var len = GW1000Api.ConvertBigEndianUInt16(data, 3); + + // Only loop as far as last record (7 bytes) minus the checksum byte + for (int i = 5; i <= len - 6; i += 7) + { + if (PrintSensorInfoNew(data, i)) + { + batteryLow = true; + } + } + + cumulus.BatteryLowAlarm.Triggered = batteryLow; + + return; + } + else + { + return; + } + } + catch (Exception ex) + { + cumulus.LogErrorMessage("GetSensorIdsNew: Unexpected error - " + ex.Message); + // no idea, so report battery as good + } + } + + private bool PrintSensorInfoNew(byte[] data, int idx) + { + // expected response + // 0 - 0xff - header + // 1 - 0xff - header + // 2 - 0x3C - sensor id command + // 3-4 - 0x?? - size of response + // 5 - wh65 0 + // 6-9 - wh65 id 1-4 + // 10 - wh65 battery 5 + // 11 - wh65 signal 0-4 6 + // 12 - wh68 + // ... etc + // (??) - 0x?? - checksum + + var batteryLow = false; + + try + { + var id = GW1000Api.ConvertBigEndianUInt32(data, idx + 1); + string type = null; + try + { + type = Enum.GetName(typeof(GW1000Api.SensorIds), data[idx]).ToUpper(); + } catch + { + // leave the device id as null + } + var battPos = idx + 5; + var sigPos = idx + 6; + if (string.IsNullOrEmpty(type)) + { + type = $"[unknown sensor = {data[idx]}]"; + } + // Wh65 could be a Wh65 or a Wh24, we found out using the System Info command + if (type == "WH65") + { + type = "WH24/WH65"; + } + + switch (id) + { + case 0xFFFFFFFE: + cumulus.LogDebugMessage($" - {type} sensor = disabled"); + return false; + case 0xFFFFFFFF: + cumulus.LogDebugMessage($" - {type} sensor = registering"); + return false; + default: + break; + } + + string batt; + double battV; + switch (type) + { + case "WH40": + // Older WH40 units do not send battery info + // Old ones report a dummy value of 16 = 1.6V + // Newer ones report volts * 100! + battV = data[battPos] / 10.0; + if (battV > 2) + { + battV /= 10.0; + batt = $"{battV:f2}V ({TestBatteryWh40(data[battPos], battV)})"; // low = 1.2V + } + else + { + batt = $"{battV:f2}V ({TestBatteryWh40(data[battPos], battV)})"; // low = 1.2V + if (battV >= 1.59) + { + batt += " dummy value?"; + } + } + + break; + + case "WH65": + case "WH24": + case "WH26": + batt = TestBattery1(data[battPos], 1); // 0 or 1 + break; + + case string wh34 when wh34.StartsWith("WH34"): // ch 1-8 + case string wh35 when wh35.StartsWith("WH35"): // ch 1-8 + case "WH90": + // if a WS90 is connected, it has a 8.8 second update rate, so reduce the MX update rate from the default 10 seconds + if (updateRate > 8000 && updateRate != 8000) + { + cumulus.LogMessage($"PrintSensorInfoNew: WS90 sensor detected, changing the update rate from {(updateRate / 1000):D} seconds to 8 seconds"); + updateRate = 8000; + } + battV = data[battPos] * 0.02; + batt = $"{battV:f2}V ({(battV > 2.4 ? "OK" : "Low")})"; + break; + + case string wh31 when wh31.StartsWith("WH31"): // ch 1-8 + batt = $"{data[battPos]} ({TestBattery1(data[battPos], 1)})"; + break; + + case "WH68": + case string wh51 when wh51.StartsWith("WH51"): // ch 1-8 + battV = data[battPos] * 0.1; + batt = $"{battV:f2}V ({TestBattery10(data[battPos])})"; // volts/10, low = 1.2V + break; + + case "WH25": + case "WH45": + case "WH57": + case string wh41 when wh41.StartsWith("WH41"): // ch 1-4 + case string wh55 when wh55.StartsWith("WH55"): // ch 1-4 + batt = $"{data[battPos]} ({TestBattery3(data[battPos])})"; // 0-5, low = 1 + break; + + case "WH80": + case "WS80": + // if a WS80 is connected, it has a 4.75 second update rate, so reduce the MX update rate from the default 10 seconds + if (updateRate > 4000 && updateRate != 4000) + { + cumulus.LogMessage($"PrintSensorInfoNew: WS80 sensor detected, changing the update rate from {(updateRate / 1000):D} seconds to 4 seconds"); + updateRate = 4000; + } + battV = data[battPos] * 0.02; + batt = $"{battV:f2}V ({(battV > 2.4 ? "OK" : "Low")})"; + break; + case "WS85": + // if a WS85 is connected, it has a 8.5 second update rate, so reduce the MX update rate from the default 10 seconds + if (updateRate > 8000 && updateRate != 8000) + { + cumulus.LogMessage($"PrintSensorInfoNew: WS85 sensor detected, changing the update rate from {(updateRate / 1000):D} seconds to 8 seconds"); + updateRate = 8000; + } + batt = $"{data[battPos]} ({TestBattery3(data[battPos])})"; // 0-5 (6+9), low = 1 + break; + default: + batt = "???"; + break; + } + + if (batt.Contains("Low")) + batteryLow = true; + + SensorReception[type] = data[sigPos]; + + cumulus.LogDebugMessage($" - {type} sensor id = {id} signal = {data[sigPos]} battery = {batt}"); + } + catch (Exception ex) + { + cumulus.LogErrorMessage("PrintSensorInfoNew: Error - " + ex.Message); + } + + return batteryLow; + } + + private void GetLiveData() + { + cumulus.LogDebugMessage("Reading live data"); + + byte[] data = Api.DoCommand(GW1000Api.Commands.CMD_GW1000_LIVEDATA); + + // sample data = in-temp, in-hum, abs-baro, rel-baro, temp, hum, dir, speed, gust, light, UV uW, UV-I, rain-rate, rain-day, rain-week, rain-month, rain-year, PM2.5, PM-ch1, Soil-1, temp-2, hum-2, temp-3, hum-3, batt + //byte[] data = new byte[] { 0xFF,0xFF,0x27,0x00,0x5D,0x01,0x00,0x83,0x06,0x55,0x08,0x26,0xE7,0x09,0x26,0xDC,0x02,0x00,0x5D,0x07,0x61,0x0A,0x00,0x89,0x0B,0x00,0x19,0x0C,0x00,0x25,0x15,0x00,0x00,0x00,0x00,0x16,0x00,0x00,0x17,0x00,0x0E,0x00,0x3C,0x10,0x00,0x1E,0x11,0x01,0x4A,0x12,0x00,0x00,0x02,0x68,0x13,0x00,0x00,0x14,0xDC,0x2A,0x01,0x90,0x4D,0x00,0xE3,0x2C,0x34,0x1B,0x00,0xD3,0x23,0x3C,0x1C,0x00,0x60,0x24,0x5A,0x4C,0x04,0x00,0x00,0x00,0xFF,0x5C,0xFF,0x00,0xF4,0xFF,0xFF,0xFF,0xFF,0xFF,0x00,0x00,0xBA } + //byte[] data = new byte[] { 0xFF, 0xFF, 0x27, 0x00, 0x6D, 0x01, 0x00, 0x96, 0x06, 0x3C, 0x08, 0x27, 0x00, 0x09, 0x27, 0x49, 0x02, 0x00, 0x16, 0x07, 0x61, 0x0A, 0x00, 0x62, 0x0B, 0x00, 0x00, 0x0C, 0x00, 0x06, 0x15, 0x00, 0x01, 0x7D, 0x40, 0x16, 0x00, 0x00, 0x17, 0x00, 0x0E, 0x00, 0x00, 0x10, 0x00, 0x00, 0x11, 0x00, 0xF7, 0x12, 0x00, 0x00, 0x01, 0x5C, 0x13, 0x00, 0x00, 0x15, 0x54, 0x2A, 0x06, 0x40, 0x4D, 0x00, 0xAB, 0x1A, 0xFF, 0x3E, 0x22, 0x39, 0x1B, 0x00, 0x3D, 0x23, 0x51, 0x1C, 0x00, 0xA0, 0x24, 0x45, 0x1D, 0x00, 0xA4, 0x25, 0x3C, 0x1E, 0x00, 0x9D, 0x26, 0x3E, 0x4C, 0x04, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xA4, 0x00, 0xF4, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x19, 0x00, 0x1A, 0x8F } + //byte[] data = new byte[] { 0xFF, 0xFF, 0x27, 0x00, 0x6D, 0x01, 0x00, 0xF8, 0x06, 0x35, 0x08, 0x27, 0xD6, 0x09, 0x27, 0xE1, 0x02, 0x00, 0xD2, 0x07, 0x5E, 0x0A, 0x00, 0x79, 0x0B, 0x00, 0x05, 0x0C, 0x00, 0x05, 0x15, 0x00, 0x00, 0x00, 0x00, 0x16, 0x00, 0x02, 0x17, 0x00, 0x2A, 0x01, 0x71, 0x4D, 0x00, 0xC4, 0x1A, 0x00, 0xE4, 0x22, 0x3B, 0x4C, 0x05, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0x06, 0xF5, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x19, 0x00, 0x33, 0x0E, 0x00, 0x00, 0x10, 0x00, 0x00, 0x11, 0x00, 0x0D, 0x12, 0x00, 0x00, 0x01, 0x0B, 0x13, 0x00, 0x00, 0x3A, 0x62, 0x0D, 0x00, 0x00, 0x70, 0x00, 0xED, 0x3A, 0x00, 0x2B, 0x00, 0x11, 0x00, 0x1E, 0x00, 0x0D, 0x03, 0x7B, 0x03, 0xD2, 0x06, 0x02 } + + // expected response + // 0 - 0xff - header + // 1 - 0xff + // 2 - 0x27 - live data command + // 3 - 0x?? - size of response1 + // 4 - 0x?? - size of response2 + // 5-X - data - NOTE format is Bigendian + // Y - 0x?? - checksum + + // See: https://osswww.ecowitt.net/uploads/20210716/WN1900%20GW1000,1100%20WH2680,2650%20telenet%20v1.6.0%20.pdf + + try + { + if (null != data && data.Length > 16) + { +#pragma warning disable S125 // Sections of code should not be commented out + /* + * debugging code with example data + * + //var hex = "FFFF27004601009D06220821A509270D02001707490A00B40B002F0C0069150001F07C16006317012A00324D00341900AA0E0000100000110000120000009D130000072C0D0000F8"; + var hex = "ffff27009c0100c806360827500927500200b107630a00710b00000c00001500000a2816000017002c2d2e28303332561c00c224361d00c325361e00c226361f00c3273621001b580059006200000012616609bbfc60011900240e00001000d31100d3120000022f130000022f0d00af6300684d6400cd4465006d4a66ff5b4e6700c74d6b00dc31002700190018001002550265000a0008002200160630"; + int NumberChars = hex.Length; + byte[] bytes = new byte[NumberChars / 2]; + for (int i = 0; i < NumberChars; i += 2) + bytes[i / 2] = Convert.ToByte(hex.Substring(i, 2), 16); + data = bytes; + */ +#pragma warning restore S125 // Sections of code should not be commented out + + // now decode it + Int16 tempInt16; + UInt16 tempUint16; + UInt32 tempUint32; + var idx = 5; + var dateTime = DateTime.Now; + var size = GW1000Api.ConvertBigEndianUInt16(data, 3); + + double windSpeedLast = -999, rainRateLast = -999, rainLast = -999, gustLast = -999; + int windDirLast = -999; + double outdoortemp = -999; + //double dewpoint + double windchill = -999; + + bool batteryLow = false; + + // We check the new value against what we have already, if older then ignore it! + double newLightningDistance = 999; + var newLightningTime = new DateTime(1900, 1, 1, 0, 0, 0, 0, DateTimeKind.Utc); + + do + { + int chan; + switch (data[idx++]) + { + case 0x01: //Indoor Temperature (℃) + tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx); + // user has mapped indoor temp to outdoor temp + if (cumulus.Gw1000PrimaryTHSensor == 99) + { + // do not process temperature here as if "MX calculates DP" is enabled, we have not yet read the humidity value. Have to do it at the end. + outdoortemp = tempInt16 / 10.0; + } + DoIndoorTemp(ConvertUnits.TempCToUser(tempInt16 / 10.0)); + idx += 2; + break; + case 0x02: //Outdoor Temperature (℃) + if (cumulus.Gw1000PrimaryTHSensor == 0) + { + tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx); + // do not process temperature here as if "MX calculates DP" is enabled, we have not yet read the humidity value. Have to do it at the end. + outdoortemp = tempInt16 / 10.0; + } + idx += 2; + break; + case 0x03: //Dew point (℃) + //tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx) + //dewpoint = tempInt16 / 10.0 + idx += 2; + break; + case 0x04: //Wind chill (℃) + if (cumulus.Gw1000PrimaryTHSensor == 0) + { + tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx); + windchill = tempInt16 / 10.0; + } + idx += 2; + break; + case 0x05: //Heat index (℃) + // cumulus calculates this + idx += 2; + break; + case 0x06: //Indoor Humidity(%) + // user has mapped indoor hum to outdoor hum + if (cumulus.Gw1000PrimaryTHSensor == 99) + { + DoOutdoorHumidity(data[idx], dateTime); + } + DoIndoorHumidity(data[idx]); + idx += 1; + break; + case 0x07: //Outdoor Humidity (%) + if (cumulus.Gw1000PrimaryTHSensor == 0) + { + DoOutdoorHumidity(data[idx], dateTime); + } + idx += 1; + break; + case 0x08: //Absolute Barometric (hPa) + tempUint16 = GW1000Api.ConvertBigEndianUInt16(data, idx); + StationPressure = ConvertUnits.PressMBToUser(tempUint16 / 10.0); + AltimeterPressure = ConvertUnits.PressMBToUser(MeteoLib.StationToAltimeter(tempUint16 / 10.0, AltitudeM(cumulus.Altitude))); + // Leave calculate SLP until the end as it depends on temperature + idx += 2; + break; + case 0x09: //Relative Barometric (hPa) + if (!cumulus.StationOptions.CalculateSLP) + { + tempUint16 = GW1000Api.ConvertBigEndianUInt16(data, idx); + DoPressure(ConvertUnits.PressMBToUser(tempUint16 / 10.0), dateTime); + } + idx += 2; + break; + case 0x0A: //Wind Direction (360°) + windDirLast = GW1000Api.ConvertBigEndianUInt16(data, idx); + idx += 2; + break; + case 0x0B: //Wind Speed (m/s) + windSpeedLast = ConvertUnits.WindMSToUser(GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0); + idx += 2; + break; + case 0x0C: // Gust speed (m/s) + gustLast = ConvertUnits.WindMSToUser(GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0); + idx += 2; + break; + case 0x0D: //Rain Event (mm) + if (cumulus.Gw1000PrimaryRainSensor == 0) + { + StormRain = ConvertUnits.RainMMToUser(GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0); + } + idx += 2; + break; + case 0x0E: //Rain Rate (mm/h) + if (cumulus.Gw1000PrimaryRainSensor == 0) + { + rainRateLast = ConvertUnits.RainMMToUser(GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0); + } + idx += 2; + break; + case 0x0F: //Rain Gain (mm) + idx += 2; + break; + case 0x10: //Rain Day (mm) + idx += 2; + break; + case 0x11: //Rain Week (mm) + idx += 2; + break; + case 0x12: //Rain Month (mm) + idx += 4; + break; + case 0x13: //Rain Year (mm) + if (cumulus.Gw1000PrimaryRainSensor == 0) + { + rainLast = ConvertUnits.RainMMToUser(GW1000Api.ConvertBigEndianUInt32(data, idx) / 10.0); + } + idx += 4; + break; + case 0x14: //Rain Totals (mm) + idx += 4; + break; + case 0x15: //Light (lux) + // Save the Lux value + LightValue = GW1000Api.ConvertBigEndianUInt32(data, idx) / 10.0; + // convert Lux to W/m² - approximately! + DoSolarRad((int) (LightValue * cumulus.SolarOptions.LuxToWM2), dateTime); + idx += 4; + break; + case 0x16: //UV (µW/cm²) - what use is this! + idx += 2; + break; + case 0x17: //UVI (0-15 index) + DoUV(data[idx], dateTime); + idx += 1; + break; + case 0x18: //Date and time + // does not appear to be implemented + idx += 6; + break; + case 0x19: //Day max wind(m/s) + idx += 2; + break; + case 0x1A: //Temperature 1(℃) + case 0x1B: //Temperature 2(℃) + case 0x1C: //Temperature 3(℃) + case 0x1D: //Temperature 4(℃) + case 0x1E: //Temperature 5(℃) + case 0x1F: //Temperature 6(℃) + case 0x20: //Temperature 7(℃) + case 0x21: //Temperature 8(℃) + chan = data[idx - 1] - 0x1A + 1; + tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx); + if (cumulus.Gw1000PrimaryTHSensor == chan) + { + outdoortemp = tempInt16 / 10.0; + } + DoExtraTemp(ConvertUnits.TempCToUser(tempInt16 / 10.0), chan); + idx += 2; + break; + case 0x22: //Humidity 1, 0-100% + case 0x23: //Humidity 2, 0-100% + case 0x24: //Humidity 3, 0-100% + case 0x25: //Humidity 4, 0-100% + case 0x26: //Humidity 5, 0-100% + case 0x27: //Humidity 6, 0-100% + case 0x28: //Humidity 7, 0-100% + case 0x29: //Humidity 8, 0-100% + chan = data[idx - 1] - 0x22 + 1; + if (cumulus.Gw1000PrimaryTHSensor == chan) + { + DoOutdoorHumidity(data[idx], dateTime); + } + DoExtraHum(data[idx], chan); + idx += 1; + break; + case 0x2B: //Soil Temperature1 (℃) + case 0x2D: //Soil Temperature2 (℃) + case 0x2F: //Soil Temperature3 (℃) + case 0x31: //Soil Temperature4 (℃) + case 0x33: //Soil Temperature5 (℃) + case 0x35: //Soil Temperature6 (℃) + case 0x37: //Soil Temperature7 (℃) + case 0x39: //Soil Temperature8 (℃) + case 0x3B: //Soil Temperature9 (℃) + case 0x3D: //Soil Temperature10 (℃) + case 0x3F: //Soil Temperature11 (℃) + case 0x41: //Soil Temperature12 (℃) + case 0x43: //Soil Temperature13 (℃) + case 0x45: //Soil Temperature14 (℃) + case 0x47: //Soil Temperature15 (℃) + case 0x49: //Soil Temperature16 (℃) + // figure out the channel number + chan = data[idx - 1] - 0x2B + 2; // -> 2,4,6,8... + chan /= 2; // -> 1,2,3,4... + tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx); + DoSoilTemp(ConvertUnits.TempCToUser(tempInt16 / 10.0), chan); + idx += 2; + break; + case 0x2C: //Soil Moisture1 (%) + case 0x2E: //Soil Moisture2 (%) + case 0x30: //Soil Moisture3 (%) + case 0x32: //Soil Moisture4 (%) + case 0x34: //Soil Moisture5 (%) + case 0x36: //Soil Moisture6 (%) + case 0x38: //Soil Moisture7 (%) + case 0x3A: //Soil Moisture8 (%) + case 0x3C: //Soil Moisture9 (%) + case 0x3E: //Soil Moisture10 (%) + case 0x40: //Soil Moisture11 (%) + case 0x42: //Soil Moisture12 (%) + case 0x44: //Soil Moisture13 (%) + case 0x46: //Soil Moisture14 (%) + case 0x48: //Soil Moisture15 (%) + case 0x4A: //Soil Moisture16 (%) + // figure out the channel number + chan = data[idx - 1] - 0x2C + 2; // -> 2,4,6,8... + chan /= 2; // -> 1,2,3,4... + DoSoilMoisture(data[idx], chan); + idx += 1; + break; + case 0x4C: //All sensor lowbatt 16 char + // This has been deprecated since v1.6.5 - now use CMD_READ_SENSOR_ID_NEW + idx += 16; + break; + case 0x2A: //PM2.5 Air Quality Sensor(μg/m³) + tempUint16 = GW1000Api.ConvertBigEndianUInt16(data, idx); + DoAirQuality(tempUint16 / 10.0, 1); + idx += 2; + break; + case 0x4D: //for pm25_ch1 + case 0x4E: //for pm25_ch2 + case 0x4F: //for pm25_ch3 + case 0x50: //for pm25_ch4 + chan = data[idx - 1] - 0x4D + 1; + tempUint16 = GW1000Api.ConvertBigEndianUInt16(data, idx); + DoAirQualityAvg(tempUint16 / 10.0, chan); + idx += 2; + break; + case 0x51: //PM2.5 ch_2 Air Quality Sensor(μg/m³) + case 0x52: //PM2.5 ch_3 Air Quality Sensor(μg/m³) + case 0x53: //PM2.5 ch_4 Air Quality Sensor(μg/m³) + chan = data[idx - 1] - 0x51 + 2; + tempUint16 = GW1000Api.ConvertBigEndianUInt16(data, idx); + DoAirQuality(tempUint16 / 10.0, chan); + idx += 2; + break; + case 0x58: //Leak ch1 + case 0x59: //Leak ch2 + case 0x5A: //Leak ch3 + case 0x5B: //Leak ch4 + chan = data[idx - 1] - 0x58 + 1; + DoLeakSensor(data[idx], chan); + idx += 1; + break; + case 0x60: //Lightning dist (1-40km) + // Sends a default value of 255km until the first strike is detected + newLightningDistance = data[idx] == 0xFF ? 999 : ConvertUnits.KmtoUserUnits(data[idx]); + idx += 1; + break; + case 0x61: //Lightning time (UTC) + // Sends a default value until the first strike is detected of 0xFFFFFFFF + tempUint32 = GW1000Api.ConvertBigEndianUInt32(data, idx); + if (tempUint32 == 0xFFFFFFFF) + { + newLightningTime = new DateTime(1900, 1, 1, 0, 0, 0, 0, DateTimeKind.Utc); + } + else + { + var dtDateTime = DateTime.UnixEpoch; + dtDateTime = dtDateTime.AddSeconds(tempUint32).ToLocalTime(); + newLightningTime = dtDateTime; + } + idx += 4; + break; + case 0x62: //Lightning strikes today + tempUint32 = GW1000Api.ConvertBigEndianUInt32(data, idx); + if (tempUint32 == 0 && dateTime.Minute == 59 && dateTime.Hour == 23) + { + // Ecowitt clock drift - if the count resets in the minute before midnight, ignore it until after midnight + } + else + { + LightningStrikesToday = (int) tempUint32; + } + idx += 4; + break; + // user temp = WH34 8 channel Soil or Water temperature sensors + case 0x63: // user temp ch1 (°C) + case 0x64: // user temp ch2 (°C) + case 0x65: // user temp ch3 (°C) + case 0x66: // user temp ch4 (°C) + case 0x67: // user temp ch5 (°C) + case 0x68: // user temp ch6 (°C) + case 0x69: // user temp ch7 (°C) + case 0x6A: // user temp ch8 (°C) + chan = data[idx - 1] - 0x63 + 1; + tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx); + if (cumulus.EcowittMapWN34[chan] == 0) // false = user temp, true = soil temp + { + DoUserTemp(ConvertUnits.TempCToUser(tempInt16 / 10.0), chan); + } + else + { + DoSoilTemp(ConvertUnits.TempCToUser(tempInt16 / 10.0), cumulus.EcowittMapWN34[chan]); + } + // Firmware version 1.5.9 uses 2 data bytes, 1.6.0+ uses 3 data bytes + if ((deviceModel.StartsWith("GW1000") && fwVersion >= new Version("1.6.0")) || !deviceModel.StartsWith("GW1000")) + { + var volts = TestBattery10V(data[idx + 2]); + if (volts <= 1.2) + { + batteryLow = true; + cumulus.LogWarningMessage($"WN34 channel #{chan} battery LOW = {volts}V"); + } + else + { + cumulus.LogDebugMessage($"WN34 channel #{chan} battery OK = {volts}V"); + } + idx += 3; + } + else + { + idx += 2; + } + break; + case 0x6B: //WH34 User temperature battery (8 channels) - No longer used in firmware 1.6.0+ + //Later version from ??? send an extended WH45 CO₂ data block + if (deviceModel.StartsWith("GW1000") && fwVersion < new Version("1.6.0")) + { + batteryLow = batteryLow || DoWH34BatteryStatus(data, idx); + idx += 8; + } + else + { + batteryLow = batteryLow || DoCO2DecodeNew(data, idx); + idx += 23; + } + break; + case 0x6C: // Heap size - has constant offset of +3692 to GW1100 HTTP value???? + StationFreeMemory = (int) GW1000Api.ConvertBigEndianUInt32(data, idx); + idx += 4; + break; + case 0x70: // WH45 CO₂ + batteryLow = batteryLow || DoCO2Decode(data, idx); + idx += 16; + break; + case 0x71: // Ambient ONLY - AQI + // Not doing anything with this yet + //idx += 2; // SEEMS TO BE VARIABLE?! + cumulus.LogDebugMessage("Found a device 0x71 - Ambient AQI. No decode for this yet"); + // We will have lost our place now, so bail out + idx = size; + break; + case 0x72: // WH35 Leaf Wetness ch1 + case 0x73: // WH35 Leaf Wetness ch2 + case 0x74: // WH35 Leaf Wetness ch3 + case 0x75: // WH35 Leaf Wetness ch4 + case 0x76: // WH35 Leaf Wetness ch5 + case 0x77: // WH35 Leaf Wetness ch6 + case 0x78: // WH35 Leaf Wetness ch7 + case 0x79: // WH35 Leaf Wetness ch8 + chan = data[idx - 1] - 0x72 + 2; // -> 2,4,6,8... + chan /= 2; // -> 1,2,3,4... + DoLeafWetness(data[idx], chan); + idx += 1; + break; + case 0x7A: // Rain Priority + idx += 1; + break; + case 0x7B: // Radiation compensation + idx += 1; + break; + case 0x80: // Piezo Rain Rate + if (cumulus.Gw1000PrimaryRainSensor == 1) + { + rainRateLast = ConvertUnits.RainMMToUser(GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0); + } + idx += 2; + break; + case 0x81: // Piezo Rain Event + if (cumulus.Gw1000PrimaryRainSensor == 1) + { + StormRain = ConvertUnits.RainMMToUser(GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0); + } + idx += 2; + break; + case 0x82: // Piezo Hourly Rain (not used) + idx += 2; + break; + case 0x83: // Piezo Daily Rain + idx += 2; + break; + case 0x84: // Piezo Weekly Rain + idx += 2; + break; + case 0x85: // Piezo Monthly Rain + idx += 4; + break; + case 0x86: // Piezo Yearly Rain + if (cumulus.Gw1000PrimaryRainSensor == 1) + { + rainLast = ConvertUnits.RainMMToUser(GW1000Api.ConvertBigEndianUInt32(data, idx) / 10.0); + } + idx += 4; + break; + case 0x87: // Piezo Gain - doc says size = 2*10 ? + idx += 20; + break; + case 0x88: // Piezo Rain Reset Time + idx += 3; + break; + default: + cumulus.LogDebugMessage($"Error: Unknown sensor id found = {data[idx - 1]}, at position = {idx - 1}"); + // We will have lost our place now, so bail out + idx = size; + break; + } + } while (idx < size); + + // Some debugging info + cumulus.LogDebugMessage($"LiveData: Wind Decode >> Last={windSpeedLast:F1}, LastDir={windDirLast}, Gust={gustLast:F1}, (MXAvg={WindAverage:F1})"); + + // Now do the stuff that requires more than one input parameter + + // Only set the lightning time/distance if it is newer than what we already have - the GW1000 seems to reset this value + if (newLightningTime > LightningTime) + { + LightningTime = newLightningTime; + if (newLightningDistance < 999) + LightningDistance = newLightningDistance; + } + + // Process outdoor temperature here, as GW1000 currently does not supply Dew Point so we have to calculate it in DoOutdoorTemp() + if (outdoortemp > -999) + DoOutdoorTemp(ConvertUnits.TempCToUser(outdoortemp), dateTime); + + // Same for extra T/H sensors + for (var i = 1; i <= 8; i++) + { + if (ExtraHum[i] > 0) + { + var dp = MeteoLib.DewPoint(ConvertUnits.UserTempToC(ExtraTemp[i]), ExtraHum[i]); + ExtraDewPoint[i] = ConvertUnits.TempCToUser(dp); + } + } + + if (gustLast > -999 && windSpeedLast > -999 && windDirLast > -999) + { + DoWind(gustLast, windDirLast, windSpeedLast, dateTime); + } + + if (rainLast > -999 && rainRateLast > -999) + { + DoRain(rainLast, rainRateLast, dateTime); + } + + if (outdoortemp > -999) + { + DoWindChill(windchill, dateTime); + DoApparentTemp(dateTime); + DoFeelsLike(dateTime); + DoHumidex(dateTime); + DoCloudBaseHeatIndex(dateTime); + + if (cumulus.StationOptions.CalculateSLP) + { + var abs = cumulus.Calib.Press.Calibrate(StationPressure); + var slp = MeteoLib.GetSeaLevelPressure(AltitudeM(cumulus.Altitude), ConvertUnits.UserPressToMB(abs), ConvertUnits.UserTempToC(OutdoorTemperature), cumulus.Latitude); + DoPressure(ConvertUnits.PressMBToUser(slp), dateTime); + } + } + + DoForecast("", false); + + cumulus.BatteryLowAlarm.Triggered = batteryLow; + + UpdateStatusPanel(dateTime); + UpdateMQTT(); + + dataReceived = true; + DataStopped = false; + cumulus.DataStoppedAlarm.Triggered = false; + } + else + { + cumulus.LogWarningMessage("GetLiveData: Invalid response"); + } + } + catch (Exception ex) + { + cumulus.LogErrorMessage("GetLiveData: Error - " + ex.Message); + } + } + + private void GetSystemInfo(bool driftOnly) + { + cumulus.LogMessage("Reading Ecowitt system info"); + + var data = Api.DoCommand(GW1000Api.Commands.CMD_READ_SSSS); + + // expected response + // 0 - 0xff - header + // 1 - 0xff - header + // 2 - 0x30 - system info + // 3 - 0x?? - size of response + // 4 - frequency - 0=433, 1=868MHz, 2=915MHz, 3=920MHz + // 5 - sensor type - 0=WH24, 1=WH65 + // 6-9 - UTC time + // 10 - time zone index (?) + // 11 - DST 0-1 - false/true + // 12 - 0x?? - checksum + + var now = DateTime.Now; + + if (data == null) + { + cumulus.LogWarningMessage("Nothing returned from System Info!"); + return; + } + + if (data.Length != 13) + { + cumulus.LogWarningMessage("Unexpected response to System Info!"); + return; + } + try + { + string freq; + if (data[4] == 0) + freq = "433MHz"; + else if (data[4] == 1) + freq = "868MHz"; + else if (data[4] == 2) + freq = "915MHz"; + else if (data[4] == 3) + freq = "920MHz"; + else + freq = $"Unknown [{data[4]}]"; + + var mainSensor = data[5] == 0 ? "WH24" : "Other than WH24"; + + var unix = (long) GW1000Api.ConvertBigEndianUInt32(data, 6); + var autoDST = data[11] != 0; + + // Ecowitt do not understand Unix time and add the local TZ offset and DST to it! + var offset = autoDST ? (int) TimeZoneInfo.Local.GetUtcOffset(now).TotalSeconds : (int) TimeZoneInfo.Local.BaseUtcOffset.TotalSeconds; + + var date = Utils.FromUnixTime(unix - offset); + var clockdiff = now.ToUnixTime() - (unix - offset); + + string slowfast; + + if (clockdiff == 0) + slowfast = "off"; + else if (clockdiff > 0) + slowfast = "slow"; + else + slowfast = "fast"; + + if (!driftOnly) + cumulus.LogMessage($"Gateway Info: frequency: {freq}, main sensor: {mainSensor}, date/time: {date:yyyy-MM-dd HH:mm:ss}, Automatic DST adjustment: {autoDST}"); + + cumulus.LogMessage($"Gateway Info: Gateway clock is {Math.Abs(clockdiff)} secs {slowfast} compared to Cumulus"); + } + catch (Exception ex) + { + cumulus.LogErrorMessage("Error processing System Info: " + ex.Message); + } + } + + private void GetPiezoRainData() + { + cumulus.LogDebugMessage("GetPiezoRainData: Reading piezo rain data"); + + var data = Api.DoCommand(GW1000Api.Commands.CMD_READ_RAIN); + + + // expected response - units mm + // 0 - 0xff - header + // 1 - 0xff - header + // 2 - 0x57 - rain data + // 3-4 - size(2) + // + // Field Value + // - data size + // + // 0E = rain rate + // - data(2) + // 10 = rain day + // - data(4) + // 11 = rain week + // - data(4) + // 12 = rain month + // - data(4) + // 13 = rain year + // - data(4) + // 0D = rain event + // - data(2) + // 0F = rain gain + // - data(2) + // 80 = piezo rain rate + // - data(2) + // 83 = piezo rain day + // - data(4) + // 84 = piezo rain week + // - data(4) + // 85 = piezo rain month + // - data(4) + // 86 = piezo rain year + // - data(4) + // 81 = piezo rain event + // - data(2) + // 87 = piezo gain 0-9 + // - data(2x10) + // 88 = rain reset time (hr, day [sun-0], month [jan=0]) + // - data(3) + // 7A = primary rain selection (0=No sensor, 1=Tipper, 2=Piezo) + // - data(1) + // 7B = solar gain compensation + // - data(1) + // 85 - checksum + + //data = new byte[] { 0xFF, 0xFF, 0x57, 0x00, 0x54, 0x0E, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x11, 0x00, 0x00, 0x00, 0x00, 0x12, 0x00, 0x00, 0x02, 0xF2, 0x13, 0x00, 0x00, 0x0B, 0x93, 0x0D, 0x00, 0x00, 0x0F, 0x00, 0x64, 0x80, 0x00, 0x00, 0x83, 0x00, 0x00, 0x00, 0x00, 0x84, 0x00, 0x00, 0x00, 0x00, 0x85, 0x00, 0x00, 0x01, 0xDE, 0x86, 0x00, 0x00, 0x0B, 0xF2, 0x81, 0x00, 0x00, 0x87, 0x00, 0x64, 0x00, 0x64, 0x00, 0x64, 0x00, 0x64, 0x00, 0x64, 0x00, 0x64, 0x00, 0x64, 0x00, 0x64, 0x00, 0x64, 0x00, 0x64, 0x88, 0x00, 0x00, 0x00, 0xF7 } + + if (data == null) + { + cumulus.LogErrorMessage("GetPiezoRainData: Nothing returned from Read Rain!"); + return; + } + + if (data.Length < 8) + { + cumulus.LogErrorMessage("GetPiezoRainData: Unexpected response to Read Rain!"); + return; + } + try + { + // There are reports of different sized messages from different gateways, + // so parse it sequentially like the live data rather than using fixed offsets + var idx = 5; + var size = GW1000Api.ConvertBigEndianUInt16(data, 3); + double? rRate = null; + double? rain = null; + + do + { + switch (data[idx++]) + { + // all the two byte values we are ignoring + case 0x0E: // rain rate + case 0x0D: // rain event + case 0x0F: // rain gain + case 0x81: // piezo rain event + idx += 2; + break; + // all the four byte values we are ignoring + case 0x10: // rain day + case 0x11: // rain week + case 0x12: // rain month + case 0x13: // rain year + case 0x83: // piezo rain day + case 0x84: // piezo rain week + case 0x85: // piezo rain month + idx += 4; + break; + case 0x80: // piezo rain rate + if (cumulus.StationOptions.UseRainForIsRaining == 2 && cumulus.Gw1000PrimaryRainSensor != 1) + { + IsRaining = GW1000Api.ConvertBigEndianUInt16(data, idx) > 0; + cumulus.IsRainingAlarm.Triggered = IsRaining; + } + else + { + rRate = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; + } + idx += 2; + break; + case 0x86: // piezo rain year + if (cumulus.Gw1000PrimaryRainSensor == 1) + rain = GW1000Api.ConvertBigEndianUInt32(data, idx) / 10.0; + idx += 4; + break; + case 0x87: // piezo gain 0-9 + idx += 20; + break; + case 0x88: // rain reset time +#if DEBUG + cumulus.LogDebugMessage($"GetPiezoRainData: Rain reset times - hour:{data[idx++]}, day:{data[idx++]}, month:{data[idx++]}"); +#else + idx += 3; +#endif + break; + case 0x7A: // Preferred rain sensor on station + var sensor = data[idx++]; +#if DEBUG + if (sensor == 0) + cumulus.LogDebugMessage("GetPiezoRainData: No rain sensor available"); + else if (sensor == 1) + cumulus.LogDebugMessage("GetPiezoRainData: Traditional rain sensor selected"); + else if (sensor == 2) + cumulus.LogDebugMessage("GetPiezoRainData: Piezo rain sensor selected"); + else + cumulus.LogDebugMessage("GetPiezoRainData: Unkown rain sensor selection value = " + sensor); +#endif + break; + case 0x7B: // Solar gain compensation +#if DEBUG + cumulus.LogDebugMessage($"GetPiezoRainData: Solar gain compensation = {(data[idx] == '0' ? "disabled" : "enabled")}"); +#endif + idx += 1; + break; + default: + cumulus.LogDebugMessage($"GetPiezoRainData: Error: Unknown value type found = {data[idx - 1]}, at position = {idx - 1}"); + // We will have lost our place now, so bail out + idx = size; + break; + } + + } while (idx < size); + + if (cumulus.Gw1000PrimaryRainSensor == 1) + { + if (rRate.HasValue && rain.HasValue) + { +#if DEBUG + cumulus.LogDebugMessage($"GetPiezoRainData: Rain Year: {rain:f1} mm, Rate: {rRate:f1} mm/hr"); +#endif + DoRain(ConvertUnits.RainMMToUser(rain.Value), ConvertUnits.RainMMToUser(rRate.Value), DateTime.Now); + } + else + { + cumulus.LogErrorMessage("GetPiezoRainData: Error, no piezo rain data found in the response"); + } + } + } + catch (Exception ex) + { + cumulus.LogErrorMessage("GetPiezoRainData: Error processing Rain Info: " + ex.Message); + } + } + + + private bool DoCO2Decode(byte[] data, int index) + { + bool batteryLow = false; + int idx = index; + cumulus.LogDebugMessage("WH45 CO₂: Decoding..."); + + try + { + CO2_temperature = ConvertUnits.TempCToUser(GW1000Api.ConvertBigEndianInt16(data, idx) / 10.0); + idx += 2; + CO2_humidity = data[idx++]; + CO2_pm10 = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; + idx += 2; + CO2_pm10_24h = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; + idx += 2; + CO2_pm2p5 = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; + idx += 2; + CO2_pm2p5_24h = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; + idx += 2; + CO2 = GW1000Api.ConvertBigEndianUInt16(data, idx); + idx += 2; + CO2_24h = GW1000Api.ConvertBigEndianUInt16(data, idx); + idx += 2; + var batt = TestBattery3(data[idx]); + var msg = $"WH45 CO₂: temp={CO2_temperature.ToString(cumulus.TempFormat)}, hum={CO2_humidity}, pm10={CO2_pm10:F1}, pm10_24h={CO2_pm10_24h:F1}, pm2.5={CO2_pm2p5:F1}, pm2.5_24h={CO2_pm2p5_24h:F1}, CO₂={CO2}, CO₂_24h={CO2_24h}"; + if (batt == "Low") + { + batteryLow = true; + msg += $", Battery={batt}"; + } + else + { + msg += $", Battery={batt}"; + } + cumulus.LogDebugMessage(msg); + } + catch (Exception ex) + { + cumulus.LogErrorMessage("DoCO2Decode: Error - " + ex.Message); + } + + return batteryLow; + } + + private bool DoCO2DecodeNew(byte[] data, int index) + { + bool batteryLow = false; + int idx = index; + cumulus.LogDebugMessage("WH45 CO₂ New: Decoding..."); + + try + { + CO2_temperature = ConvertUnits.TempCToUser(GW1000Api.ConvertBigEndianInt16(data, idx) / 10.0); + idx += 2; + CO2_humidity = data[idx++]; + CO2_pm10 = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; + idx += 2; + CO2_pm10_24h = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; + idx += 2; + CO2_pm2p5 = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; + idx += 2; + CO2_pm2p5_24h = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; + idx += 2; + CO2 = GW1000Api.ConvertBigEndianUInt16(data, idx); + idx += 2; + CO2_24h = GW1000Api.ConvertBigEndianUInt16(data, idx); + idx += 2; + CO2_pm1 = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; + idx += 2; + CO2_pm1_24h = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; + idx += 2; + CO2_pm4 = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; + idx += 2; + CO2_pm4_24h = GW1000Api.ConvertBigEndianUInt16(data, idx)/ 10.0; + idx += 2; + var msg = $"WH45 CO₂ New: temp={CO2_temperature.ToString(cumulus.TempFormat)}, hum={CO2_humidity}, pm10={CO2_pm10:F1}, pm10_24h={CO2_pm10_24h:F1}, pm2.5={CO2_pm2p5:F1}, pm2.5_24h={CO2_pm2p5_24h:F1}, CO₂={CO2}, CO₂_24h={CO2_24h}, pm1={CO2_pm1:F1}, pm1_24h={CO2_pm1_24h:F1}, pm4={CO2_pm4:F1}, pm4_24h={CO2_pm4_24h:F1}"; + var batt = TestBattery3(data[idx]); + batteryLow = batt == "Low"; + msg += $", Battery={batt}"; + cumulus.LogDebugMessage(msg); + + CO2_pm2p5_aqi = GetAqi(AqMeasure.pm2p5, CO2_pm2p5); + CO2_pm2p5_24h_aqi = GetAqi(AqMeasure.pm2p5h24, CO2_pm2p5_24h); + CO2_pm10_aqi = GetAqi(AqMeasure.pm10, CO2_pm10); + CO2_pm10_24h_aqi = GetAqi(AqMeasure.pm10h24, CO2_pm10_24h); + } + catch (Exception ex) + { + cumulus.LogExceptionMessage(ex, "DoCO2DecodeNew: Error"); + } + return batteryLow; + } + + private bool DoWH34BatteryStatus(byte[] data, int index) + { + // No longer used in firmware 1.6.0+ + cumulus.LogDebugMessage("WH34 battery status..."); + var str = "wh34>" + + " ch1=" + TestBattery3(data[index + 1]) + + " ch2=" + TestBattery3(data[index + 2]) + + " ch3=" + TestBattery3(data[index + 3]) + + " ch4=" + TestBattery3(data[index + 4]) + + " ch5=" + TestBattery3(data[index + 5]) + + " ch6=" + TestBattery3(data[index + 6]) + + " ch7=" + TestBattery3(data[index + 7]) + + " ch8=" + TestBattery3(data[index + 8]); + + cumulus.LogDebugMessage(str); + + return str.Contains("Low"); + } + + private static string TestBattery1(byte value, byte mask) + { + return (value & mask) == 0 ? "OK" : "Low"; + } + + private static string TestBattery3(byte value) + { + if (value == 6) + return "DC"; + if (value == 9) + return "OFF"; + return value > 1 ? "OK" : "Low"; + } + + private static string TestBattery10(byte value) + { + // consider 1.2V as low + if (value == 255) + return "n/a"; + + return value > 12 ? "OK" : "Low"; + } + + private static double TestBattery10V(byte value) + { + return value / 10.0; + } + + private static string TestBatteryWh40(byte value, double volts) + { + if (value == 255) + return "n/a"; + + return volts > 1.2 ? "OK" : "Low"; + } + + + private sealed class Discovery + { + public List IP { get; set; } + public List Name { get; set; } + public List Mac { get; set; } + + public Discovery() + { + IP = []; + Name = []; + Mac = []; + } + } + + private void DataTimeout(object source, ElapsedEventArgs e) + { + if (dataReceived) + { + dataReceived = false; + DataStopped = false; + cumulus.DataStoppedAlarm.Triggered = false; + } + else + { + cumulus.LogErrorMessage($"ERROR: No data received from the GW1000 for {tmrDataWatchdog.Interval / 1000} seconds"); + if (!DataStopped) + { + DataStoppedTime = DateTime.Now; + DataStopped = true; + } + cumulus.DataStoppedAlarm.LastMessage = $"No data received from the GW1000 for {tmrDataWatchdog.Interval / 1000} seconds"; + cumulus.DataStoppedAlarm.Triggered = true; + DoDiscovery(); + PostDiscovery(); + } + } + } +} diff --git a/CumulusMX/EcowittLocalApi.cs b/CumulusMX/EcowittLocalApi.cs new file mode 100644 index 00000000..9e7fc4d3 --- /dev/null +++ b/CumulusMX/EcowittLocalApi.cs @@ -0,0 +1,168 @@ +using System; +using System.Net.Sockets; + +using static MailKit.Telemetry; + +namespace CumulusMX +{ + internal sealed class EcowittLocalApi : IDisposable + { + private readonly Cumulus cumulus; + private string ipAddress = null; + + public EcowittLocalApi(Cumulus cumul) + { + cumulus = cumul; + } + + + public void GetLiveData() + { + // http://ip-address/get_livedata_info + // + // Returns an almighty mess! They couldn't have made this any worse if they tried! + // All values are returned as strings - including integers and decimals + // Some values include the units in the value string, others have a separate field for the unit + // The separate sensors return an arrays that only ever contain a single object + // + // { + // "common_list": [ + // { + // id: "ITEM_ID", + // val: "VALUE[ [UNIT]]", + // unit: "UNIT", + // [Battery: "VALUE"] + // }, + // {etc} + // ], + // "rain": [ + // { + // id: "ITEM_ID", + // val: "VALUE[ [UNIT]]", + // unit: "UNIT", + // [Battery: "VALUE"] + // }, + // {etc} + // ], + // "piezoRain": [ + // { + // id: "ITEM_ID", + // val: "[DECIMAL][ [UNIT]]", + // unit: "UNIT", + // [Battery: "[INT]" + // }, + // {etc} + // ], + // "wh25": [ + // { + // "intemp": "[DECIMAL]", + // "unit": "C|F", + // "inhumi": "[INT]%", + // "abs": "[DECIMAL] [hPa|???]", + // "rel": "[DECIMAL] [hPa|???]", + // "battery": "[INT]" + // } + // ], + // "lightning": [ + // { + // "distance": "[INT] [km|???]", + // "timestamp": "MM/DD/YYYY HH:MM:SS", + // "count": "[INT]", + // "battery": "[INT]" + // } + // ], + // "co2": [ + // { + // "temp": "[DECIMAL]", + // "unit": "[C|F]", + // "humidity": "[INT]%", + // "PM25": "[DECIMAL]", + // "PM25_RealAQI": "[INT]", + // "PM25_24HAQI": "[INT]", + // "PM10": "[DECIMAL]", + // "PM10_RealAQI": "[INT]", + // "PM10_24HAQI": "[INT]", + // "CO2": "[INT]", + // "CO2_24H": "[INT]", + // "battery": "[INT]" + // } + // ], + // "ch_pm25": [ + // { + // "channel": "[1-4]", + // "PM25": "[DECIMAL]", + // "PM25_RealAQI": "[INT]", + // "PM25_24HAQI": "[INT]", + // "battery": "[INT]" + // }, + // {etc} + // ], + // "ch_aisle": [ + // { + // "channel": "[1-16]", + // "name": "", + // "battery": "[INT]", + // "temp": "[DECIMAL]", + // "unit": "[C|F]", + // "humidity": "[[INT]%|None]" + // }, + // {etc} + // ], + // "ch_soil": [ + // { + // "channel": "[1-16]", + // "name": "", + // "battery": "[INT", + // "humidity": "[INT]%" + // }, + // {etc} + // ], + // "ch_temp": [ + // { + // "channel": "[1-16]", + // "name": "", + // "temp": "[DECIMAL]", + // "unit": "[C|F]", + // "battery": "[INT]" + // }, + // {etc} + // ], + // "ch_leaf": [ + // { + // "channel": "[1-??]", + // "name": "", + // "humidity": "[INT]%" + // "battery": "[INT]", + // }, + // {etc} + // ] + // } + // + // + // Sample: + // {"common_list": [{"id": "0x02", "val": "23.5", "unit": "C"}, {"id": "0x07", "val": "57%"}, {"id": "3", "val": "23.5", "unit": "C"}, {"id": "0x03", "val": "14.5", "unit": "C"}, {"id": "0x0B", "val": "9.00 km/h"}, {"id": "0x0C", "val": "9.00 km/h"}, {"id": "0x19", "val": "26.64 km/h"}, {"id": "0x15", "val": "646.57 W/m2"}, {"id": "0x17", "val": "3"}, {"id": "0x0A", "val": "295"}], "rain": [{"id": "0x0D", "val": "0.0 mm"}, {"id": "0x0E", "val": "0.0 mm/Hr"}, {"id": "0x10", "val": "0.0 mm"}, {"id": "0x11", "val": "5.0 mm"}, {"id": "0x12", "val": "27.1 mm"}, {"id": "0x13", "val": "681.4 mm", "battery": "0"}], "piezoRain": [{"id": "0x0D", "val": "0.0 mm"}, {"id": "0x0E", "val": "0.0 mm/Hr"}, {"id": "0x10", "val": "0.0 mm"}, {"id": "0x11", "val": "10.7 mm"}, {"id": "0x12", "val": "32.3 mm"}, {"id": "0x13", "val": "678.3 mm", "battery": "5"}], "wh25": [{"intemp": "26.0", "unit": "C", "inhumi": "56%", "abs": "993.0 hPa", "rel": "1027.4 hPa", "battery": "0"}], "lightning": [{"distance": "12 km", "timestamp": "07/15/2024 20: 46: 42", "count": "0", "battery": "3"}], "co2": [{"temp": "24.4", "unit": "C", "humidity": "62%", "PM25": "0.9", "PM25_RealAQI": "4", "PM25_24HAQI": "7", "PM10": "0.9", "PM10_RealAQI": "1", "PM10_24HAQI": "2", "CO2": "323", "CO2_24H": "348", "battery": "6"}], "ch_pm25": [{"channel": "1", "PM25": "6.0", "PM25_RealAQI": "25", "PM25_24HAQI": "24", "battery": "5"}, {"channel": "2", "PM25": "8.0", "PM25_RealAQI": "33", "PM25_24HAQI": "32", "battery": "5"}], "ch_leak": [{"channel": "2", "name": "", "battery": "4", "status": "Normal"}], "ch_aisle": [{"channel": "1", "name": "", "battery": "0", "temp": "24.9", "unit": "C", "humidity": "61%"}, {"channel": "2", "name": "", "battery": "0", "temp": "25.7", "unit": "C", "humidity": "64%"}, {"channel": "3", "name": "", "battery": "0", "temp": "23.6", "unit": "C", "humidity": "63%"}, {"channel": "4", "name": "", "battery": "0", "temp": "34.9", "unit": "C", "humidity": "83%"}, {"channel": "5", "name": "", "battery": "0", "temp": "-14.4", "unit": "C", "humidity": "None"}, {"channel": "6", "name": "", "battery": "0", "temp": "31.5", "unit": "C", "humidity": "56%"}, {"channel": "7", "name": "", "battery": "0", "temp": "8.2", "unit": "C", "humidity": "50%"}], "ch_soil": [{"channel": "1", "name": "", "battery": "5", "humidity": "56%"}, {"channel": "2", "name": "", "battery": "4", "humidity": "47%"}, {"channel": "3", "name": "", "battery": "5", "humidity": "27%"}, {"channel": "4", "name": "", "battery": "5", "humidity": "50%"}, {"channel": "5", "name": "", "battery": "4", "humidity": "54%"}, {"channel": "6", "name": "", "battery": "4", "humidity": "47%"}], "ch_temp": [{"channel": "1", "name": "", "temp": "21.5", "unit": "C", "battery": "3"}, {"channel": "2", "name": "", "temp": "16.4", "unit": "C", "battery": "5"}], "ch_leaf": [{"channel": "1", "name": "CH1 Leaf Wetness", "humidity": "10%", "battery": "5"}]} + + } + + + + public void GetSensorInfo() + { + // http://ip-address/get_sensors_info?page=1 + // http://ip-address/get_sensors_info?page=2 + + } + + + public void Dispose() + { + try + { + } + catch + { + // do nothing + } + } + } +} From 32914473b4cae9822cffd48ae611510da10d3ae1 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Thu, 22 Aug 2024 20:19:44 +0100 Subject: [PATCH 07/63] version uplift --- CumulusMX/CumulusMX.csproj | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CumulusMX/CumulusMX.csproj b/CumulusMX/CumulusMX.csproj index 3a33a2dc..6a70abb0 100644 --- a/CumulusMX/CumulusMX.csproj +++ b/CumulusMX/CumulusMX.csproj @@ -55,7 +55,7 @@ - 4.2.0.4029 + 4.2.0.4030 Copyright © 2015-$([System.DateTime]::Now.ToString('yyyy')) Cumulus MX From b8b6bb695dbeefca99f2bd393573f94414aea955 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sun, 25 Aug 2024 16:58:03 +0100 Subject: [PATCH 08/63] More work on Ecowitt Local HTTP API --- CumulusMX/EcowittHttpApiStation.cs | 502 +++-------------------------- CumulusMX/EcowittLocalApi.cs | 232 ++++++++++++- CumulusMX/WeatherStation.cs | 5 +- 3 files changed, 280 insertions(+), 459 deletions(-) diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index f2afbe1a..b8a4a5b6 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -25,7 +25,7 @@ internal class EcowittHttpApiStation : WeatherStation private int lastHour = -1; private readonly EcowittApi ecowittApi; - private readonly GW1000Api Api; + private readonly EcowittLocalApi localApi; private int maxArchiveRuns = 1; @@ -40,7 +40,7 @@ internal class EcowittHttpApiStation : WeatherStation internal static readonly char[] dotSeparator = ['.']; internal static readonly string[] underscoreV = ["_V"]; - public GW1000Station(Cumulus cumulus) : base(cumulus) + public EcowittHttpApiStation(Cumulus cumulus) : base(cumulus) { cumulus.Units.AirQualityUnitText = "µg/m³"; cumulus.Units.SoilMoistureUnitText = "%"; @@ -102,12 +102,11 @@ public GW1000Station(Cumulus cumulus) : base(cumulus) ipaddr = cumulus.Gw1000IpAddress; macaddr = cumulus.Gw1000MacAddress; - Api = new GW1000Api(cumulus); + localApi = new EcowittLocalApi(cumulus); ecowittApi = new EcowittApi(cumulus, this); DoDiscovery(); - PostDiscovery(); _ = CheckAvailableFirmware(); @@ -132,7 +131,7 @@ public override void Start() DoDayResetIfNeeded(); DoTrendValues(DateTime.Now); - cumulus.LogMessage("Starting Ecowitt Local API"); + cumulus.LogMessage("Starting Ecowitt Local HTTP API"); cumulus.StartTimersAndSensors(); @@ -146,82 +145,60 @@ public override void Start() while (!cumulus.cancellationToken.IsCancellationRequested) { - if (Api.Connected) + var rawData = localApi.GetLiveData(cumulus.cancellationToken); + dataLastRead = DateTime.Now; + + // every 30 seconds read the rain rate + if ((cumulus.Gw1000PrimaryRainSensor == 1 || cumulus.StationOptions.UseRainForIsRaining == 2) && (DateTime.UtcNow - piezoLastRead).TotalSeconds >= 30 && !cumulus.cancellationToken.IsCancellationRequested) + { + GetPiezoRainData(); + piezoLastRead = DateTime.UtcNow; + } + + var minute = DateTime.Now.Minute; + if (minute != lastMinute) { - GetLiveData(); - dataLastRead = DateTime.Now; + lastMinute = minute; - // every 30 seconds read the rain rate - if ((cumulus.Gw1000PrimaryRainSensor == 1 || cumulus.StationOptions.UseRainForIsRaining == 2) && (DateTime.UtcNow - piezoLastRead).TotalSeconds >= 30 && !cumulus.cancellationToken.IsCancellationRequested) + // at the start of every 20 minutes to trigger battery status check + if ((minute % 20) == 0 && !cumulus.cancellationToken.IsCancellationRequested) { - GetPiezoRainData(); - piezoLastRead = DateTime.UtcNow; + GetSensorIdsNew(); } - var minute = DateTime.Now.Minute; - if (minute != lastMinute) + // every day dump the clock drift at midday each day + if (minute == 0 && DateTime.Now.Hour == 12) { - lastMinute = minute; - - // at the start of every 20 minutes to trigger battery status check - if ((minute % 20) == 0 && !cumulus.cancellationToken.IsCancellationRequested) - { - GetSensorIdsNew(); - } + GetSystemInfo(true); + } - // every day dump the clock drift at midday each day - if (minute == 0 && DateTime.Now.Hour == 12) - { - GetSystemInfo(true); - } + var hour = DateTime.Now.Hour; + if (lastHour != hour) + { + lastHour = hour; - var hour = DateTime.Now.Hour; - if (lastHour != hour) + if (hour == 13) { - lastHour = hour; - - if (hour == 13) + var fw = GetFirmwareVersion(); + if (fw != "???") { - var fw = GetFirmwareVersion(); - if (fw != "???") - { - GW1000FirmwareVersion = fw; - deviceModel = GW1000FirmwareVersion.Split('_')[0]; - deviceFirmware = GW1000FirmwareVersion.Split('_')[1]; + GW1000FirmwareVersion = fw; + deviceModel = GW1000FirmwareVersion.Split('_')[0]; + deviceFirmware = GW1000FirmwareVersion.Split('_')[1]; - var fwString = GW1000FirmwareVersion.Split(underscoreV, StringSplitOptions.None); - if (fwString.Length > 1) - { - fwVersion = new Version(fwString[1]); - } - else - { - // failed to get the version, lets assume it's fairly new - fwVersion = new Version("1.6.5"); - } + var fwString = GW1000FirmwareVersion.Split(underscoreV, StringSplitOptions.None); + if (fwString.Length > 1) + { + fwVersion = new Version(fwString[1]); + } + else + { + // failed to get the version, lets assume it's fairly new + fwVersion = new Version("1.6.5"); } - - _ = CheckAvailableFirmware(); } - } - } - } - else - { - cumulus.LogMessage("Attempting to reconnect to Ecowitt device..."); - Api.OpenTcpPort(cumulus.Gw1000IpAddress, AtPort); - if (Api.Connected) - { - cumulus.LogMessage("Reconnected to Ecowitt device"); - GetLiveData(); - } - else - { - // add a small extra delay before trying again - cumulus.LogMessage("Delaying before attempting reconnect"); - if (cumulus.cancellationToken.WaitHandle.WaitOne(TimeSpan.FromMilliseconds(20000))) - { - break; + + _ = CheckAvailableFirmware(); } } } @@ -241,7 +218,6 @@ public override void Start() } finally { - Api.CloseTcpPort(); cumulus.LogMessage("Local API task ended"); } }, cumulus.cancellationToken); @@ -556,67 +532,9 @@ private void DoDiscovery() return; } - private void PostDiscovery() - { - cumulus.LogMessage("Using IP address = " + ipaddr + " Port = " + AtPort); - - Api.OpenTcpPort(ipaddr, AtPort); - - if (Api.Connected) - { - cumulus.LogMessage("Connected OK"); - Cumulus.LogConsoleMessage("Connected to station", ConsoleColor.White, true); - } - else - { - cumulus.LogErrorMessage("Not Connected"); - Cumulus.LogConsoleMessage("Unable to connect to station", ConsoleColor.Red, true); - } - - if (Api.Connected) - { - // Get the firmware version as check we are communicating - GW1000FirmwareVersion = GetFirmwareVersion(); - cumulus.LogMessage($"Ecowitt firmware version: {GW1000FirmwareVersion}"); - if (GW1000FirmwareVersion != "???") - { - deviceModel = GW1000FirmwareVersion.Split('_')[0]; - deviceFirmware = GW1000FirmwareVersion.Split('_')[1]; - - var fwString = GW1000FirmwareVersion.Split(underscoreV, StringSplitOptions.None); - if (fwString.Length > 1) - { - fwVersion = new Version(fwString[1]); - } - else - { - // failed to get the version, lets assume it's fairly new - fwVersion = new Version("1.6.5"); - } - } - - GetSystemInfo(false); - - GetSensorIdsNew(); - } - } - private string GetFirmwareVersion() { var response = "???"; - cumulus.LogMessage("Reading firmware version"); - try - { - var data = Api.DoCommand(GW1000Api.Commands.CMD_READ_FIRMWARE_VERSION); - if (null != data && data.Length > 0) - { - response = Encoding.ASCII.GetString(data, 5, data[4]); - } - } - catch (Exception ex) - { - cumulus.LogErrorMessage($"GetFirmwareVersion: Error retrieving/processing firmware version. Message - {ex.Message}"); - } return response; } @@ -657,205 +575,12 @@ private void GetSensorIdsNew() { cumulus.LogMessage("Reading sensor ids"); - var data = Api.DoCommand(GW1000Api.Commands.CMD_READ_SENSOR_ID_NEW); - - // expected response - // 0 - 0xff - header - // 1 - 0xff - header - // 2 - 0x3C - sensor id command - // 3-4 - 0x?? - size of response - // 5 - wh65 - // 6-9 - wh65 id - // 10 - wh65 battery - // 11 - wh65 signal 0-4 - // 12 - wh68 - // ... etc - // (??) - 0x?? - checksum - - var batteryLow = false; - - try - { - if (null != data && data.Length > 200) - { - var len = GW1000Api.ConvertBigEndianUInt16(data, 3); - - // Only loop as far as last record (7 bytes) minus the checksum byte - for (int i = 5; i <= len - 6; i += 7) - { - if (PrintSensorInfoNew(data, i)) - { - batteryLow = true; - } - } - - cumulus.BatteryLowAlarm.Triggered = batteryLow; - - return; - } - else - { - return; - } - } - catch (Exception ex) - { - cumulus.LogErrorMessage("GetSensorIdsNew: Unexpected error - " + ex.Message); - // no idea, so report battery as good - } } private bool PrintSensorInfoNew(byte[] data, int idx) { - // expected response - // 0 - 0xff - header - // 1 - 0xff - header - // 2 - 0x3C - sensor id command - // 3-4 - 0x?? - size of response - // 5 - wh65 0 - // 6-9 - wh65 id 1-4 - // 10 - wh65 battery 5 - // 11 - wh65 signal 0-4 6 - // 12 - wh68 - // ... etc - // (??) - 0x?? - checksum - var batteryLow = false; - try - { - var id = GW1000Api.ConvertBigEndianUInt32(data, idx + 1); - string type = null; - try - { - type = Enum.GetName(typeof(GW1000Api.SensorIds), data[idx]).ToUpper(); - } catch - { - // leave the device id as null - } - var battPos = idx + 5; - var sigPos = idx + 6; - if (string.IsNullOrEmpty(type)) - { - type = $"[unknown sensor = {data[idx]}]"; - } - // Wh65 could be a Wh65 or a Wh24, we found out using the System Info command - if (type == "WH65") - { - type = "WH24/WH65"; - } - - switch (id) - { - case 0xFFFFFFFE: - cumulus.LogDebugMessage($" - {type} sensor = disabled"); - return false; - case 0xFFFFFFFF: - cumulus.LogDebugMessage($" - {type} sensor = registering"); - return false; - default: - break; - } - - string batt; - double battV; - switch (type) - { - case "WH40": - // Older WH40 units do not send battery info - // Old ones report a dummy value of 16 = 1.6V - // Newer ones report volts * 100! - battV = data[battPos] / 10.0; - if (battV > 2) - { - battV /= 10.0; - batt = $"{battV:f2}V ({TestBatteryWh40(data[battPos], battV)})"; // low = 1.2V - } - else - { - batt = $"{battV:f2}V ({TestBatteryWh40(data[battPos], battV)})"; // low = 1.2V - if (battV >= 1.59) - { - batt += " dummy value?"; - } - } - - break; - - case "WH65": - case "WH24": - case "WH26": - batt = TestBattery1(data[battPos], 1); // 0 or 1 - break; - - case string wh34 when wh34.StartsWith("WH34"): // ch 1-8 - case string wh35 when wh35.StartsWith("WH35"): // ch 1-8 - case "WH90": - // if a WS90 is connected, it has a 8.8 second update rate, so reduce the MX update rate from the default 10 seconds - if (updateRate > 8000 && updateRate != 8000) - { - cumulus.LogMessage($"PrintSensorInfoNew: WS90 sensor detected, changing the update rate from {(updateRate / 1000):D} seconds to 8 seconds"); - updateRate = 8000; - } - battV = data[battPos] * 0.02; - batt = $"{battV:f2}V ({(battV > 2.4 ? "OK" : "Low")})"; - break; - - case string wh31 when wh31.StartsWith("WH31"): // ch 1-8 - batt = $"{data[battPos]} ({TestBattery1(data[battPos], 1)})"; - break; - - case "WH68": - case string wh51 when wh51.StartsWith("WH51"): // ch 1-8 - battV = data[battPos] * 0.1; - batt = $"{battV:f2}V ({TestBattery10(data[battPos])})"; // volts/10, low = 1.2V - break; - - case "WH25": - case "WH45": - case "WH57": - case string wh41 when wh41.StartsWith("WH41"): // ch 1-4 - case string wh55 when wh55.StartsWith("WH55"): // ch 1-4 - batt = $"{data[battPos]} ({TestBattery3(data[battPos])})"; // 0-5, low = 1 - break; - - case "WH80": - case "WS80": - // if a WS80 is connected, it has a 4.75 second update rate, so reduce the MX update rate from the default 10 seconds - if (updateRate > 4000 && updateRate != 4000) - { - cumulus.LogMessage($"PrintSensorInfoNew: WS80 sensor detected, changing the update rate from {(updateRate / 1000):D} seconds to 4 seconds"); - updateRate = 4000; - } - battV = data[battPos] * 0.02; - batt = $"{battV:f2}V ({(battV > 2.4 ? "OK" : "Low")})"; - break; - case "WS85": - // if a WS85 is connected, it has a 8.5 second update rate, so reduce the MX update rate from the default 10 seconds - if (updateRate > 8000 && updateRate != 8000) - { - cumulus.LogMessage($"PrintSensorInfoNew: WS85 sensor detected, changing the update rate from {(updateRate / 1000):D} seconds to 8 seconds"); - updateRate = 8000; - } - batt = $"{data[battPos]} ({TestBattery3(data[battPos])})"; // 0-5 (6+9), low = 1 - break; - default: - batt = "???"; - break; - } - - if (batt.Contains("Low")) - batteryLow = true; - - SensorReception[type] = data[sigPos]; - - cumulus.LogDebugMessage($" - {type} sensor id = {id} signal = {data[sigPos]} battery = {batt}"); - } - catch (Exception ex) - { - cumulus.LogErrorMessage("PrintSensorInfoNew: Error - " + ex.Message); - } - return batteryLow; } @@ -863,23 +588,7 @@ private void GetLiveData() { cumulus.LogDebugMessage("Reading live data"); - byte[] data = Api.DoCommand(GW1000Api.Commands.CMD_GW1000_LIVEDATA); - - // sample data = in-temp, in-hum, abs-baro, rel-baro, temp, hum, dir, speed, gust, light, UV uW, UV-I, rain-rate, rain-day, rain-week, rain-month, rain-year, PM2.5, PM-ch1, Soil-1, temp-2, hum-2, temp-3, hum-3, batt - //byte[] data = new byte[] { 0xFF,0xFF,0x27,0x00,0x5D,0x01,0x00,0x83,0x06,0x55,0x08,0x26,0xE7,0x09,0x26,0xDC,0x02,0x00,0x5D,0x07,0x61,0x0A,0x00,0x89,0x0B,0x00,0x19,0x0C,0x00,0x25,0x15,0x00,0x00,0x00,0x00,0x16,0x00,0x00,0x17,0x00,0x0E,0x00,0x3C,0x10,0x00,0x1E,0x11,0x01,0x4A,0x12,0x00,0x00,0x02,0x68,0x13,0x00,0x00,0x14,0xDC,0x2A,0x01,0x90,0x4D,0x00,0xE3,0x2C,0x34,0x1B,0x00,0xD3,0x23,0x3C,0x1C,0x00,0x60,0x24,0x5A,0x4C,0x04,0x00,0x00,0x00,0xFF,0x5C,0xFF,0x00,0xF4,0xFF,0xFF,0xFF,0xFF,0xFF,0x00,0x00,0xBA } - //byte[] data = new byte[] { 0xFF, 0xFF, 0x27, 0x00, 0x6D, 0x01, 0x00, 0x96, 0x06, 0x3C, 0x08, 0x27, 0x00, 0x09, 0x27, 0x49, 0x02, 0x00, 0x16, 0x07, 0x61, 0x0A, 0x00, 0x62, 0x0B, 0x00, 0x00, 0x0C, 0x00, 0x06, 0x15, 0x00, 0x01, 0x7D, 0x40, 0x16, 0x00, 0x00, 0x17, 0x00, 0x0E, 0x00, 0x00, 0x10, 0x00, 0x00, 0x11, 0x00, 0xF7, 0x12, 0x00, 0x00, 0x01, 0x5C, 0x13, 0x00, 0x00, 0x15, 0x54, 0x2A, 0x06, 0x40, 0x4D, 0x00, 0xAB, 0x1A, 0xFF, 0x3E, 0x22, 0x39, 0x1B, 0x00, 0x3D, 0x23, 0x51, 0x1C, 0x00, 0xA0, 0x24, 0x45, 0x1D, 0x00, 0xA4, 0x25, 0x3C, 0x1E, 0x00, 0x9D, 0x26, 0x3E, 0x4C, 0x04, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xA4, 0x00, 0xF4, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x19, 0x00, 0x1A, 0x8F } - //byte[] data = new byte[] { 0xFF, 0xFF, 0x27, 0x00, 0x6D, 0x01, 0x00, 0xF8, 0x06, 0x35, 0x08, 0x27, 0xD6, 0x09, 0x27, 0xE1, 0x02, 0x00, 0xD2, 0x07, 0x5E, 0x0A, 0x00, 0x79, 0x0B, 0x00, 0x05, 0x0C, 0x00, 0x05, 0x15, 0x00, 0x00, 0x00, 0x00, 0x16, 0x00, 0x02, 0x17, 0x00, 0x2A, 0x01, 0x71, 0x4D, 0x00, 0xC4, 0x1A, 0x00, 0xE4, 0x22, 0x3B, 0x4C, 0x05, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0x06, 0xF5, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x19, 0x00, 0x33, 0x0E, 0x00, 0x00, 0x10, 0x00, 0x00, 0x11, 0x00, 0x0D, 0x12, 0x00, 0x00, 0x01, 0x0B, 0x13, 0x00, 0x00, 0x3A, 0x62, 0x0D, 0x00, 0x00, 0x70, 0x00, 0xED, 0x3A, 0x00, 0x2B, 0x00, 0x11, 0x00, 0x1E, 0x00, 0x0D, 0x03, 0x7B, 0x03, 0xD2, 0x06, 0x02 } - - // expected response - // 0 - 0xff - header - // 1 - 0xff - // 2 - 0x27 - live data command - // 3 - 0x?? - size of response1 - // 4 - 0x?? - size of response2 - // 5-X - data - NOTE format is Bigendian - // Y - 0x?? - checksum - - // See: https://osswww.ecowitt.net/uploads/20210716/WN1900%20GW1000,1100%20WH2680,2650%20telenet%20v1.6.0%20.pdf + byte[] data = null; // localApi.DoCommand(GW1000Api.Commands.CMD_GW1000_LIVEDATA); try { @@ -1262,7 +971,7 @@ private void GetLiveData() idx += 4; break; case 0x70: // WH45 CO₂ - batteryLow = batteryLow || DoCO2Decode(data, idx); + batteryLow = batteryLow || DoCO2DecodeNew(data, idx); idx += 16; break; case 0x71: // Ambient ONLY - AQI @@ -1417,83 +1126,13 @@ private void GetSystemInfo(bool driftOnly) { cumulus.LogMessage("Reading Ecowitt system info"); - var data = Api.DoCommand(GW1000Api.Commands.CMD_READ_SSSS); - - // expected response - // 0 - 0xff - header - // 1 - 0xff - header - // 2 - 0x30 - system info - // 3 - 0x?? - size of response - // 4 - frequency - 0=433, 1=868MHz, 2=915MHz, 3=920MHz - // 5 - sensor type - 0=WH24, 1=WH65 - // 6-9 - UTC time - // 10 - time zone index (?) - // 11 - DST 0-1 - false/true - // 12 - 0x?? - checksum - - var now = DateTime.Now; - - if (data == null) - { - cumulus.LogWarningMessage("Nothing returned from System Info!"); - return; - } - - if (data.Length != 13) - { - cumulus.LogWarningMessage("Unexpected response to System Info!"); - return; - } - try - { - string freq; - if (data[4] == 0) - freq = "433MHz"; - else if (data[4] == 1) - freq = "868MHz"; - else if (data[4] == 2) - freq = "915MHz"; - else if (data[4] == 3) - freq = "920MHz"; - else - freq = $"Unknown [{data[4]}]"; - - var mainSensor = data[5] == 0 ? "WH24" : "Other than WH24"; - - var unix = (long) GW1000Api.ConvertBigEndianUInt32(data, 6); - var autoDST = data[11] != 0; - - // Ecowitt do not understand Unix time and add the local TZ offset and DST to it! - var offset = autoDST ? (int) TimeZoneInfo.Local.GetUtcOffset(now).TotalSeconds : (int) TimeZoneInfo.Local.BaseUtcOffset.TotalSeconds; - - var date = Utils.FromUnixTime(unix - offset); - var clockdiff = now.ToUnixTime() - (unix - offset); - - string slowfast; - - if (clockdiff == 0) - slowfast = "off"; - else if (clockdiff > 0) - slowfast = "slow"; - else - slowfast = "fast"; - - if (!driftOnly) - cumulus.LogMessage($"Gateway Info: frequency: {freq}, main sensor: {mainSensor}, date/time: {date:yyyy-MM-dd HH:mm:ss}, Automatic DST adjustment: {autoDST}"); - - cumulus.LogMessage($"Gateway Info: Gateway clock is {Math.Abs(clockdiff)} secs {slowfast} compared to Cumulus"); - } - catch (Exception ex) - { - cumulus.LogErrorMessage("Error processing System Info: " + ex.Message); - } } private void GetPiezoRainData() { cumulus.LogDebugMessage("GetPiezoRainData: Reading piezo rain data"); - var data = Api.DoCommand(GW1000Api.Commands.CMD_READ_RAIN); + byte[] data = null; //localApi.DoCommand(GW1000Api.Commands.CMD_READ_RAIN); // expected response - units mm @@ -1661,50 +1300,6 @@ private void GetPiezoRainData() } - private bool DoCO2Decode(byte[] data, int index) - { - bool batteryLow = false; - int idx = index; - cumulus.LogDebugMessage("WH45 CO₂: Decoding..."); - - try - { - CO2_temperature = ConvertUnits.TempCToUser(GW1000Api.ConvertBigEndianInt16(data, idx) / 10.0); - idx += 2; - CO2_humidity = data[idx++]; - CO2_pm10 = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; - idx += 2; - CO2_pm10_24h = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; - idx += 2; - CO2_pm2p5 = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; - idx += 2; - CO2_pm2p5_24h = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; - idx += 2; - CO2 = GW1000Api.ConvertBigEndianUInt16(data, idx); - idx += 2; - CO2_24h = GW1000Api.ConvertBigEndianUInt16(data, idx); - idx += 2; - var batt = TestBattery3(data[idx]); - var msg = $"WH45 CO₂: temp={CO2_temperature.ToString(cumulus.TempFormat)}, hum={CO2_humidity}, pm10={CO2_pm10:F1}, pm10_24h={CO2_pm10_24h:F1}, pm2.5={CO2_pm2p5:F1}, pm2.5_24h={CO2_pm2p5_24h:F1}, CO₂={CO2}, CO₂_24h={CO2_24h}"; - if (batt == "Low") - { - batteryLow = true; - msg += $", Battery={batt}"; - } - else - { - msg += $", Battery={batt}"; - } - cumulus.LogDebugMessage(msg); - } - catch (Exception ex) - { - cumulus.LogErrorMessage("DoCO2Decode: Error - " + ex.Message); - } - - return batteryLow; - } - private bool DoCO2DecodeNew(byte[] data, int index) { bool batteryLow = false; @@ -1843,7 +1438,6 @@ private void DataTimeout(object source, ElapsedEventArgs e) cumulus.DataStoppedAlarm.LastMessage = $"No data received from the GW1000 for {tmrDataWatchdog.Interval / 1000} seconds"; cumulus.DataStoppedAlarm.Triggered = true; DoDiscovery(); - PostDiscovery(); } } } diff --git a/CumulusMX/EcowittLocalApi.cs b/CumulusMX/EcowittLocalApi.cs index 9e7fc4d3..6283c6de 100644 --- a/CumulusMX/EcowittLocalApi.cs +++ b/CumulusMX/EcowittLocalApi.cs @@ -1,7 +1,10 @@ using System; -using System.Net.Sockets; +using System.Threading; + +using ServiceStack; +using ServiceStack.Text; + -using static MailKit.Telemetry; namespace CumulusMX { @@ -16,7 +19,7 @@ public EcowittLocalApi(Cumulus cumul) } - public void GetLiveData() + public liveData GetLiveData(CancellationToken token) { // http://ip-address/get_livedata_info // @@ -142,8 +145,52 @@ public void GetLiveData() // Sample: // {"common_list": [{"id": "0x02", "val": "23.5", "unit": "C"}, {"id": "0x07", "val": "57%"}, {"id": "3", "val": "23.5", "unit": "C"}, {"id": "0x03", "val": "14.5", "unit": "C"}, {"id": "0x0B", "val": "9.00 km/h"}, {"id": "0x0C", "val": "9.00 km/h"}, {"id": "0x19", "val": "26.64 km/h"}, {"id": "0x15", "val": "646.57 W/m2"}, {"id": "0x17", "val": "3"}, {"id": "0x0A", "val": "295"}], "rain": [{"id": "0x0D", "val": "0.0 mm"}, {"id": "0x0E", "val": "0.0 mm/Hr"}, {"id": "0x10", "val": "0.0 mm"}, {"id": "0x11", "val": "5.0 mm"}, {"id": "0x12", "val": "27.1 mm"}, {"id": "0x13", "val": "681.4 mm", "battery": "0"}], "piezoRain": [{"id": "0x0D", "val": "0.0 mm"}, {"id": "0x0E", "val": "0.0 mm/Hr"}, {"id": "0x10", "val": "0.0 mm"}, {"id": "0x11", "val": "10.7 mm"}, {"id": "0x12", "val": "32.3 mm"}, {"id": "0x13", "val": "678.3 mm", "battery": "5"}], "wh25": [{"intemp": "26.0", "unit": "C", "inhumi": "56%", "abs": "993.0 hPa", "rel": "1027.4 hPa", "battery": "0"}], "lightning": [{"distance": "12 km", "timestamp": "07/15/2024 20: 46: 42", "count": "0", "battery": "3"}], "co2": [{"temp": "24.4", "unit": "C", "humidity": "62%", "PM25": "0.9", "PM25_RealAQI": "4", "PM25_24HAQI": "7", "PM10": "0.9", "PM10_RealAQI": "1", "PM10_24HAQI": "2", "CO2": "323", "CO2_24H": "348", "battery": "6"}], "ch_pm25": [{"channel": "1", "PM25": "6.0", "PM25_RealAQI": "25", "PM25_24HAQI": "24", "battery": "5"}, {"channel": "2", "PM25": "8.0", "PM25_RealAQI": "33", "PM25_24HAQI": "32", "battery": "5"}], "ch_leak": [{"channel": "2", "name": "", "battery": "4", "status": "Normal"}], "ch_aisle": [{"channel": "1", "name": "", "battery": "0", "temp": "24.9", "unit": "C", "humidity": "61%"}, {"channel": "2", "name": "", "battery": "0", "temp": "25.7", "unit": "C", "humidity": "64%"}, {"channel": "3", "name": "", "battery": "0", "temp": "23.6", "unit": "C", "humidity": "63%"}, {"channel": "4", "name": "", "battery": "0", "temp": "34.9", "unit": "C", "humidity": "83%"}, {"channel": "5", "name": "", "battery": "0", "temp": "-14.4", "unit": "C", "humidity": "None"}, {"channel": "6", "name": "", "battery": "0", "temp": "31.5", "unit": "C", "humidity": "56%"}, {"channel": "7", "name": "", "battery": "0", "temp": "8.2", "unit": "C", "humidity": "50%"}], "ch_soil": [{"channel": "1", "name": "", "battery": "5", "humidity": "56%"}, {"channel": "2", "name": "", "battery": "4", "humidity": "47%"}, {"channel": "3", "name": "", "battery": "5", "humidity": "27%"}, {"channel": "4", "name": "", "battery": "5", "humidity": "50%"}, {"channel": "5", "name": "", "battery": "4", "humidity": "54%"}, {"channel": "6", "name": "", "battery": "4", "humidity": "47%"}], "ch_temp": [{"channel": "1", "name": "", "temp": "21.5", "unit": "C", "battery": "3"}, {"channel": "2", "name": "", "temp": "16.4", "unit": "C", "battery": "5"}], "ch_leaf": [{"channel": "1", "name": "CH1 Leaf Wetness", "humidity": "10%", "battery": "5"}]} - } + string responseBody; + int responseCode; + int retries = 3; + + string ip; + int retry = 1; + + ip = cumulus.DavisOptions.IPAddr; + + do + { + var url = $"http://{ip}/get_livedata_info?"; + + // we want to do this synchronously, so .Result + using (var response = cumulus.MyHttpClient.GetAsync(url, token).Result) + { + responseBody = response.Content.ReadAsStringAsync(token).Result; + responseCode = (int) response.StatusCode; + cumulus.LogDebugMessage($"LocalApi.GetLiveData: Ecowitt Local API GetLiveData Response code: {responseCode}"); + cumulus.LogDataMessage($"LocalApi.GetLiveData: Ecowitt Local API GetLiveData Response: {responseBody}"); + } + if (responseCode != 200) + { + cumulus.LogWarningMessage($"LocalApi.GetLiveData: Ecowitt Local API GetLiveData Error: {responseCode}"); + Cumulus.LogConsoleMessage($" - Error {responseCode}", ConsoleColor.Red); + return null; + } + + + if (responseBody == "{}") + { + cumulus.LogMessage("LocalApi.GetLiveData: Ecowitt Local API GetLiveData: No data was returned."); + Cumulus.LogConsoleMessage(" - No Live data available"); + return null; + } + else if (responseBody.StartsWith("{ \"common")) // sanity check + { + // Convert JSON string to an object + liveData json = responseBody.FromJson(); + return json; + } + } while (retries-- > 0); + + return null; + } public void GetSensorInfo() @@ -164,5 +211,182 @@ public void Dispose() // do nothing } } + + private enum commonSensorTypes + { + indoorTemp = 1, + temp = 2, + dewpoint = 3, + windchill = 4, + heatindex = 5, + indoorHum = 6, + hum = 7, + absPressure = 8, + relPressure = 9, + windDir = 10, + windSpeed = 11, + windGust = 12, + rainEvent = 13, + rainRate = 14, + rainGain = 15, + rainDay = 16, + rainWeek = 17, + rainMonth = 18, + rainYear =19, + rainTotals = 20, + light = 21, + uv = 22, + uvindex = 23, + time = 24, + dayWindMax = 25 + } + + public class commonSensor + { + public int id { get; set ; } + public string val { get; set; } + public string? unit { get; set; } + public double? battery { get; set; } + + public int? valInt + { + get + { + if (val.EndsWith('%')) + val = val[0..^1]; + return int.TryParse(val, out int result) ? result : null; + } + } + + public double? valDbl + { + get + { + if (val.Contains(' ')) + { + var temp = val.Split(' '); + unit = temp[1]; + val = temp[0]; + } + return double.TryParse(val, out double result) ? result : null; + } + } + } + + public class tempHumSensor + { + public int channel { get; set; } + public int battery { get; set; } + public double? temp { get; set; } + public string? humidity { get; set; } + public string? unit { get; set; } + + public int? humidityVal + { + get + { + if (humidity.EndsWith('%')) + humidity = humidity[0..^1]; + return int.TryParse(humidity, out int result) ? result : null; + } + } + } + + + public class wh25Sensor + { + public double intemp { get; set; } + public string unit { get; set; } + public string inhumi { get; set; } + public string abs { get; set; } + public string rel { get; set; } + public int battery { get; set; } + + public int? inhumiInt + { + get + { + return int.TryParse(inhumi[0..^1], out int result) ? result : null; + } + } + } + + public class lightningSensor + { + public string distance { get; set; } + public string timestamp { get; set; } + public int count { get; set; } + public int battery { get; set; } + + public double? distanceVal + { + get + { + var temp = distance.Split(' '); + return double.TryParse(temp[0], out double result) ? result : null; + } + } + + public string? distanceUnit + { + get + { + var temp = distance.Split(' '); + return temp[1]; + } + } + } + + public class co2Sensor + { + public double temp { get; set; } + public string unit { get; set; } + public string humidity { get; set; } + public double PM25 { get; set; } + public double PM25_RealAQI { get; set; } + public double PM25_24HAQI { get; set; } + public double PM10 { get; set; } + public double PM10_RealAQI { get; set; } + public double PM10_24HAQI { get; set; } + public int CO2 { get; set; } + public int CO2_24H { get; set; } + public int battery { get; set; } + } + + public class ch_pm25Sensor + { + public int channel { get; set; } + public double PM25 { get; set; } + public double PM25_RealAQI { get; set; } + public double PM25_24HAQI { get; set; } + public int battery { get; set; } + } + + public class ch_leakSensor + { + public int channel { get; set; } + public string name { get; set; } + public int battery { get; set; } + public string status { get; set; } + } + + public class liveData + { + public commonSensor[] common_list { get; set; } + public commonSensor[]? rain { get; set; } + public commonSensor[]? piezoRain { get; set; } + public wh25Sensor[]? wh25 { get; set; } + public lightningSensor[]? lightning { get; set; } + public co2Sensor[]? co2 { get; set; } + public ch_pm25Sensor[]? ch_pm25 { get; set; } + public ch_leakSensor[]? ch_leak { get; set; } + public tempHumSensor[]? ch_aisle { get; set; } + public tempHumSensor[]? ch_soil { get; set; } + public tempHumSensor[]? ch_temp { get; set; } + public tempHumSensor[]? ch_leaf { get; set; } + } + + } + } diff --git a/CumulusMX/WeatherStation.cs b/CumulusMX/WeatherStation.cs index 6c187167..93d3af36 100644 --- a/CumulusMX/WeatherStation.cs +++ b/CumulusMX/WeatherStation.cs @@ -1683,7 +1683,10 @@ internal async Task sendWebSocketData(bool wait = false) try { - webSocketSemaphore.Release(); + if (webSocketSemaphore.CurrentCount == 0) + { + webSocketSemaphore.Release(); + } } catch { From 6cc29403c3cae76f561d1c803710e3f49befad99 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Mon, 26 Aug 2024 17:36:40 +0100 Subject: [PATCH 09/63] Fix WMR928 indoor temperature - no conversion to user units --- CumulusMX/WMR928Station.cs | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/CumulusMX/WMR928Station.cs b/CumulusMX/WMR928Station.cs index d5c7baa6..af44bf23 100644 --- a/CumulusMX/WMR928Station.cs +++ b/CumulusMX/WMR928Station.cs @@ -498,9 +498,9 @@ private void WMR928Indoor(List buff) IndoorBattStatus = buff[3] / 16; // Extract temp (tenths of deg C) in BCD; bytes 4 (LSB) and 5 (MSB) - double temp10 = BCDchartoint(buff[4]) + (BCDchartoint(buff[5]) * 100); + double temp = ExtractTemp(buff[4], buff[5]); - DoIndoorTemp(temp10 / 10); + DoIndoorTemp(ConvertUnits.TempCToUser(temp)); // humidity in BCD; byte 6 DoIndoorHumidity(BCDchartoint(buff[6])); @@ -557,7 +557,7 @@ private void WMR928Indoor2(List buff) // Extract temp (tenths of deg C) in BCD; double temp = ExtractTemp(buff[4], buff[5]); - DoIndoorTemp(temp); + DoIndoorTemp(ConvertUnits.TempCToUser(temp)); // humidity in BCD; byte 6 DoIndoorHumidity(BCDchartoint(buff[6])); From 498d6d1866cfd5ba5d1a9b3b131622f4b1344c44 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Mon, 26 Aug 2024 22:56:38 +0100 Subject: [PATCH 10/63] Implement GetIntervalData --- CumulusMX/Api.cs | 4 + CumulusMX/WeatherStation.cs | 214 ++++++++++++++++++++++++++++++++++++ 2 files changed, 218 insertions(+) diff --git a/CumulusMX/Api.cs b/CumulusMX/Api.cs index 9676c6ee..bc246948 100644 --- a/CumulusMX/Api.cs +++ b/CumulusMX/Api.cs @@ -239,6 +239,7 @@ public async Task GetData(string req) int start = Convert.ToInt32(query["start"]); int length = Convert.ToInt32(query["length"]); string search = query["search[value]"]; + string data = query["data"]; using var writer = HttpContext.OpenResponseText(new UTF8Encoding(false)); switch (req) @@ -273,6 +274,9 @@ public async Task GetData(string req) await writer.WriteAsync(Cumulus.GetErrorLog()); } break; + case "intervaldata.json": + await writer.WriteAsync(Station.GetIntervalData(from, to, data)); + break; default: Response.StatusCode = 404; break; diff --git a/CumulusMX/WeatherStation.cs b/CumulusMX/WeatherStation.cs index 93d3af36..36146d29 100644 --- a/CumulusMX/WeatherStation.cs +++ b/CumulusMX/WeatherStation.cs @@ -12254,6 +12254,220 @@ public string GetLogfile(string from, string to, string draw, int start, int len } + public string GetIntervalData(string from, string to, string vars) + { + var fromDate = Utils.FromUnixTime(long.Parse(from)); + var toDate = Utils.FromUnixTime(long.Parse(to)); + var variables = vars.Split(','); + + if (variables.Length == 0) + { + return "[]"; + } + + + // adjust for 9 am rollover + var ts = fromDate.AddHours(-cumulus.GetHourInc(fromDate)); + var te = toDate.AddHours(-cumulus.GetHourInc(toDate)); + var fileDate = new DateTime(ts.Year, ts.Month, 15, 0, 0, 0, DateTimeKind.Local); + + var data = new Dictionary>(); + + // add the headers + data.Add("Date/Time", variables.ToList()); + + // convert field names to numbers + var logFields = new List(); + if (variables.Contains("temp")) logFields.Add(2); + if (variables.Contains("intemp")) logFields.Add(12); + if (variables.Contains("heatindex")) logFields.Add(16); + if (variables.Contains("dewpoint")) logFields.Add(4); + if (variables.Contains("windchill")) logFields.Add(15); + if (variables.Contains("feelslike")) logFields.Add(27); + if (variables.Contains("humidex")) logFields.Add(28); + if (variables.Contains("hum")) logFields.Add(3); + if (variables.Contains("inhum")) logFields.Add(13); + if (variables.Contains("press")) logFields.Add(10); + if (variables.Contains("windspeed")) logFields.Add(5); + if (variables.Contains("windgust")) logFields.Add(6); + if (variables.Contains("winddir")) logFields.Add(7); + if (variables.Contains("rain")) logFields.Add(9); + if (variables.Contains("rainrate")) logFields.Add(8); + if (variables.Contains("solar")) logFields.Add(18); + if (variables.Contains("uv")) logFields.Add(17); + + var extraFields = new List(); + for (var i = 0; i < 10; i++) + { + if (variables.Contains($"extratemp{i}")) extraFields.Add(i + 2); + } + for (var i = 0; i < 10; i++) + { + if (variables.Contains($"extrahum{i}")) extraFields.Add(i + 12); + } + for (var i = 0; i < 10; i++) + { + if (variables.Contains($"extradew{i}")) extraFields.Add(i + 22); + } + for (var i = 0; i < 4; i++) + { + if (variables.Contains($"soiltemp{i}")) extraFields.Add(i + 32); + } + for (var i = 4; i < 16; i++) + { + if (variables.Contains($"soiltemp{i}")) extraFields.Add(i + 44); + } + for (var i = 0; i < 4; i++) + { + if (variables.Contains($"soilmoist{i}")) extraFields.Add(i + 36); + } + for (var i = 4; i < 16; i++) + { + if (variables.Contains($"soilmoist{i}")) extraFields.Add(i + 56); + } + for (var i = 0; i < 2; i++) + { + if (variables.Contains($"leafwet{i}")) extraFields.Add(i + 44); + } + for (var i = 0; i < 8; i++) + { + if (variables.Contains($"usertemp{i}")) extraFields.Add(i + 76); + } + for (var i = 0; i < 4; i++) + { + if (variables.Contains($"aqpm{i}")) extraFields.Add(i + 68); + } + for (var i = 0; i < 4; i++) + { + if (variables.Contains($"aqpmavg{i}")) extraFields.Add(i + 72); + } + if (variables.Contains("co2")) extraFields.Add(84); + if (variables.Contains("co2avg")) extraFields.Add(85); + if (variables.Contains("co2pm25")) extraFields.Add(86); + if (variables.Contains("co2pm25avg")) extraFields.Add(87); + if (variables.Contains("co2pm10")) extraFields.Add(88); + if (variables.Contains("co2pm10avg")) extraFields.Add(89); + if (variables.Contains("co2temp")) extraFields.Add(90); + if (variables.Contains("co2hum")) extraFields.Add(91); + + + + var finished = false; + var json = new StringBuilder("[", 5 * variables.Length); + + + try + { + while (!finished) + { + + + if (logFields.Count > 0) + { + var logfile = cumulus.GetLogFileName(fileDate); + + if (!File.Exists(logfile)) + { + cumulus.LogErrorMessage($"GetIntervalData: Error, file does not exist: {logfile}"); + return "[]"; + } + + cumulus.LogDebugMessage($"GetIntervalData: Processing log file - {logfile}"); + + // read the log file into a List + var lines = File.ReadAllLines(logfile).ToList(); + + foreach (var line in lines) + { + var fieldList = new List(); + var fields = line.Split(','); + var date = Utils.ddmmyyhhmmStrToDate(fields[0], fields[1]); + var dateStr = fields[0] + " " + fields[1]; + + if (date >= fromDate && date <= toDate) + { + foreach (var fieldNo in logFields) + { + fieldList.Add(fields[fieldNo]); + } + + data.TryAdd(dateStr, fieldList); + } + } + } + + if (extraFields.Count > 0) + { + var logfile = cumulus.GetExtraLogFileName(ts); + + if (!File.Exists(logfile)) + { + cumulus.LogErrorMessage($"GetIntervalData: Error, file does not exist: {logfile}"); + return "[]"; + } + + cumulus.LogDebugMessage($"GetIntervalData: Processing extra log file - {logfile}"); + + // read the log file into a List + var lines = File.ReadAllLines(logfile).ToList(); + + foreach (var line in lines) + { + var fieldList = new List(); + var fields = line.Split(','); + var date = Utils.ddmmyyhhmmStrToDate(fields[0], fields[1]); + var dateStr = fields[0] + " " + fields[1]; + + if (date >= fromDate && date <= toDate) + { + foreach (var fieldNo in extraFields) + { + fieldList.Add(fields[fieldNo]); + } + + if (data.TryGetValue(dateStr, out var value)) + { + value.AddRange(fieldList); + } + else + { + data.TryAdd(dateStr, fieldList); + } + } + } + } + + // might need the next months log + fileDate = fileDate.AddMonths(1); + + // have we run out of log entries? + // filedate is 15th on month, compare against the first + if (te <= fileDate.AddDays(-14)) + { + finished = true; + cumulus.LogDebugMessage("GetIntervalData: Finished processing log files"); + } + } + + + foreach (var rec in data) + { + json.Append($"[\"{rec.Key}\",\"{string.Join($"\",\"", rec.Value)}\"],"); + } + + json.Length -= 1; + json.Append(']'); + + return json.ToString(); + } + catch (Exception ex) + { + cumulus.LogErrorMessage("GetIntervalData: Error - " + ex.ToString()); + } + + return "[]"; + } + public string GetCachedSqlCommands(string draw, int start, int length, string search) { try From 2e7a4dc635b2f01d7522d8ec12a66a195fcaaa4b Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Mon, 26 Aug 2024 23:06:58 +0100 Subject: [PATCH 11/63] Updates.txt --- Updates.txt | 2 ++ 1 file changed, 2 insertions(+) diff --git a/Updates.txt b/Updates.txt index 3f9e26c9..791e49fc 100644 --- a/Updates.txt +++ b/Updates.txt @@ -2,12 +2,14 @@ ————————————— New - Exposes the UseDataLogger setting in Station Settings > Options > Advanced +- Implements a new data viewer where you can select and view historic data from the monthly logs for a given period, for a set of data values. See: "Data logs > Interval data viewer" Changed Fixed - Davis Cloud stations no longer continuosly try to fetch history data if there is no Pro subscription +- WMR928 Station now correctly converts indoor temperatures to the user defined units From 039891edbb8719ef017fcf6cafc7cb4bac6b3450 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Tue, 27 Aug 2024 21:16:45 +0100 Subject: [PATCH 12/63] Update Interval data Add daily data --- CumulusMX/Api.cs | 3 + CumulusMX/Cumulus.cs | 13 +++ CumulusMX/WeatherStation.cs | 221 +++++++++++++++++++++--------------- Updates.txt | 3 +- 4 files changed, 147 insertions(+), 93 deletions(-) diff --git a/CumulusMX/Api.cs b/CumulusMX/Api.cs index bc246948..1d92700a 100644 --- a/CumulusMX/Api.cs +++ b/CumulusMX/Api.cs @@ -277,6 +277,9 @@ public async Task GetData(string req) case "intervaldata.json": await writer.WriteAsync(Station.GetIntervalData(from, to, data)); break; + case "dailydata.json": + await writer.WriteAsync(Station.GetDailylData(from, to, data)); + break; default: Response.StatusCode = 404; break; diff --git a/CumulusMX/Cumulus.cs b/CumulusMX/Cumulus.cs index 91b7b36c..ca90bb20 100644 --- a/CumulusMX/Cumulus.cs +++ b/CumulusMX/Cumulus.cs @@ -349,6 +349,7 @@ public struct MqttConfig public bool EnableInterval { get; set; } public string IntervalTemplate { get; set; } } + internal MqttConfig MQTT; // NOAA report settings @@ -495,6 +496,18 @@ public struct MqttConfig internal string[] APRSstationtype = ["DsVP", "DsVP", "WMR928", "WM918", "EW", "FO", "WS2300", "FOs", "WMR100", "WMR200", "IMET", "DsVP", "Ecow", "Unkn", "Ecow", "Ambt", "Tmpt", "Simul", "Ecow", "DsVP", "DsVP", "Json"]; + internal string[] DayfileFieldNames = ["Date", "HighWindGust", "HighGustBearing", "HighGustTime", "MinTemperature", "MinTempTime", "MaxTemperature", "MaxTempTime", "MinPressure", "MinPressureTime", "MaxPressure", "MaxPressureTime", "MaxRainfallRate", "MaxRainRateTime", "TotalRainfallToday", "AvgTemperatureToday", "TotalWindRun", "HighAverageWindSpeed", "HighAvgWindSpeedTime", "LowHumidity", "LowHumidityTime", "HighHumidity", "HighHumidityTime", "TotalEvapotranspiration", "TotalHoursOfSunshine", "HighHeatIndex", "HighHeatIndexTime", "HighApparentTemperature", "HighAppTempTime", "LowApparentTemperature", "LowAppTempTime", "High1hRain", "High1hRainTime", "LowWindChill", "LowWindChillTime", "HighDewPoint", "HighDewPointTime", "LowDewPoint", "LowDewPointTime", "DominantWindBearing", "HeatingDegreeDays", "CoolingDegreeDays", "HighSolarRad", "HighSolarRadTime", "HighUv-I", "HighUv-ITime", "HighFeelsLike", "HighFeelsLikeTime", "LowFeelsLike", "LowFeelsLikeTime", "HighHumidex", "HighHumidexTime", "ChillHours", "High24hRain", "High24hRainTime"]; + internal string[] LogFileFieldNames = ["Date", "Time", "Temperature", "Humidity", "DewPoint", "WindSpeed", "RecentHighGust", "AverageWindBearing", "RainfallRate", "RainfallSoFar", "SeaLevelPressure", "RainfallCounter", "InsideTemperature", "InsideHumidity", "CurrentGust", "WindChill", "HeatIndex", "UvIndex", "SolarRadiation", "Evapotranspiration", "AnnualEvapotranspiration", "ApparentTemperature", "MaxSolarRadiation", "HoursOfSunshine", "WindBearing", "Rg-11Rain", "RainSinceMidnight", "FeelsLike", "Humidex"]; + internal string[] ExtraFileFieldNames = ["Date", "Time", + "Temp1", "Temp2", "Temp3", "Temp4", "Temp5", "Temp6", "Temp7", "Temp8", "Temp9", "Temp10", "Hum1", "Hum2", "Hum3", "Hum4", "Hum5", "Hum6", "Hum7", "Hum8", "Hum9", "Hum10", + "Dewpoint1", "Dewpoint2", "Dewpoint3", "Dewpoint4", "Dewpoint5", "Dewpoint6", "Dewpoint7", "Dewpoint8", "Dewpoint9", "Dewpoint10", + "SoilTemp1", "SoilTemp2", "SoilTemp3", "SoilTemp4", "SoilMoist1", "SoilMoist2", "SoilMoist3", "SoilMoist4", "na1", "na2", "LeafWet1", "LeafWet2", + "SoilTemp5", "SoilTemp6", "SoilTemp7", "SoilTemp8", "SoilTemp9", "SoilTemp10", "SoilTemp11", "SoilTemp12", "SoilTemp13", "SoilTemp14", "SoilTemp15", "SoilTemp16", + "SoilMoist5", "SoilMoist6", "SoilMoist7", "SoilMoist8", "SoilMoist9", "SoilMoist10", "SoilMoist11", "SoilMoist12", "SoilMoist13", "SoilMoist14", "SoilMoist15", "SoilMoist16", + "AQ1Pm", "AQ2Pm", "AQ3Pm", "AQ4Pm", "AQ1PmAvg", "AQ2PmAvg", "AQ3PmAvg", "AQ4PmAvg", "UserTemp1", "UserTemp2", "UserTemp3", "UserTemp4", "UserTemp5", "UserTemp6", "UserTemp7", "UserTemp8", + "CO2", "CO2Avg", "CO2Pm25", "CO2Pm25Avg", "CO2Pm10", "CO2Pm10Avg", "CO2Temp", "CO2Hum" + ]; + private string loggingfile; private static readonly Queue queue = new(50); public static Queue ErrorList diff --git a/CumulusMX/WeatherStation.cs b/CumulusMX/WeatherStation.cs index 36146d29..7238dd6e 100644 --- a/CumulusMX/WeatherStation.cs +++ b/CumulusMX/WeatherStation.cs @@ -12254,13 +12254,13 @@ public string GetLogfile(string from, string to, string draw, int start, int len } - public string GetIntervalData(string from, string to, string vars) + public string GetIntervalData(string from, string to, string fields) { var fromDate = Utils.FromUnixTime(long.Parse(from)); var toDate = Utils.FromUnixTime(long.Parse(to)); - var variables = vars.Split(','); + var flds = (fields ?? "").Split(',').Select(int.Parse).ToArray(); - if (variables.Length == 0) + if (flds.Length == 0) { return "[]"; } @@ -12274,95 +12274,41 @@ public string GetIntervalData(string from, string to, string vars) var data = new Dictionary>(); // add the headers - data.Add("Date/Time", variables.ToList()); + var fldNames = new List(); + var useLogFile = false; + var useExtraFile = false; - // convert field names to numbers - var logFields = new List(); - if (variables.Contains("temp")) logFields.Add(2); - if (variables.Contains("intemp")) logFields.Add(12); - if (variables.Contains("heatindex")) logFields.Add(16); - if (variables.Contains("dewpoint")) logFields.Add(4); - if (variables.Contains("windchill")) logFields.Add(15); - if (variables.Contains("feelslike")) logFields.Add(27); - if (variables.Contains("humidex")) logFields.Add(28); - if (variables.Contains("hum")) logFields.Add(3); - if (variables.Contains("inhum")) logFields.Add(13); - if (variables.Contains("press")) logFields.Add(10); - if (variables.Contains("windspeed")) logFields.Add(5); - if (variables.Contains("windgust")) logFields.Add(6); - if (variables.Contains("winddir")) logFields.Add(7); - if (variables.Contains("rain")) logFields.Add(9); - if (variables.Contains("rainrate")) logFields.Add(8); - if (variables.Contains("solar")) logFields.Add(18); - if (variables.Contains("uv")) logFields.Add(17); - - var extraFields = new List(); - for (var i = 0; i < 10; i++) - { - if (variables.Contains($"extratemp{i}")) extraFields.Add(i + 2); - } - for (var i = 0; i < 10; i++) - { - if (variables.Contains($"extrahum{i}")) extraFields.Add(i + 12); - } - for (var i = 0; i < 10; i++) - { - if (variables.Contains($"extradew{i}")) extraFields.Add(i + 22); - } - for (var i = 0; i < 4; i++) - { - if (variables.Contains($"soiltemp{i}")) extraFields.Add(i + 32); - } - for (var i = 4; i < 16; i++) - { - if (variables.Contains($"soiltemp{i}")) extraFields.Add(i + 44); - } - for (var i = 0; i < 4; i++) - { - if (variables.Contains($"soilmoist{i}")) extraFields.Add(i + 36); - } - for (var i = 4; i < 16; i++) - { - if (variables.Contains($"soilmoist{i}")) extraFields.Add(i + 56); - } - for (var i = 0; i < 2; i++) - { - if (variables.Contains($"leafwet{i}")) extraFields.Add(i + 44); - } - for (var i = 0; i < 8; i++) - { - if (variables.Contains($"usertemp{i}")) extraFields.Add(i + 76); - } - for (var i = 0; i < 4; i++) + try { - if (variables.Contains($"aqpm{i}")) extraFields.Add(i + 68); + foreach (var fld in flds) + { + if (fld < 1000) + { + fldNames.Add(cumulus.LogFileFieldNames[fld]); + useLogFile = true; + } + else + { + fldNames.Add(cumulus.ExtraFileFieldNames[fld - 1000]); + useExtraFile = true; + } + } + data.Add("Date/Time", fldNames); } - for (var i = 0; i < 4; i++) + catch (Exception ex) { - if (variables.Contains($"aqpmavg{i}")) extraFields.Add(i + 72); + cumulus.LogErrorMessage("GetIntervalData: Error processing input fields: " + ex.Message); + return "[]"; } - if (variables.Contains("co2")) extraFields.Add(84); - if (variables.Contains("co2avg")) extraFields.Add(85); - if (variables.Contains("co2pm25")) extraFields.Add(86); - if (variables.Contains("co2pm25avg")) extraFields.Add(87); - if (variables.Contains("co2pm10")) extraFields.Add(88); - if (variables.Contains("co2pm10avg")) extraFields.Add(89); - if (variables.Contains("co2temp")) extraFields.Add(90); - if (variables.Contains("co2hum")) extraFields.Add(91); - - var finished = false; - var json = new StringBuilder("[", 5 * variables.Length); - + var json = new StringBuilder("[", flds.Length * 512); try { while (!finished) { - - - if (logFields.Count > 0) + if (useLogFile) { var logfile = cumulus.GetLogFileName(fileDate); @@ -12379,16 +12325,20 @@ public string GetIntervalData(string from, string to, string vars) foreach (var line in lines) { - var fieldList = new List(); - var fields = line.Split(','); - var date = Utils.ddmmyyhhmmStrToDate(fields[0], fields[1]); - var dateStr = fields[0] + " " + fields[1]; + var vars = line.Split(','); + var date = Utils.ddmmyyhhmmStrToDate(vars[0], vars[1]); + var dateStr = vars[0] + " " + vars[1]; if (date >= fromDate && date <= toDate) { - foreach (var fieldNo in logFields) + var fieldList = new List(); + + foreach (var indx in flds) { - fieldList.Add(fields[fieldNo]); + if (indx < 1000) + { + fieldList.Add(vars[indx]); + } } data.TryAdd(dateStr, fieldList); @@ -12396,7 +12346,7 @@ public string GetIntervalData(string from, string to, string vars) } } - if (extraFields.Count > 0) + if (useExtraFile) { var logfile = cumulus.GetExtraLogFileName(ts); @@ -12414,15 +12364,18 @@ public string GetIntervalData(string from, string to, string vars) foreach (var line in lines) { var fieldList = new List(); - var fields = line.Split(','); - var date = Utils.ddmmyyhhmmStrToDate(fields[0], fields[1]); - var dateStr = fields[0] + " " + fields[1]; + var vars = line.Split(','); + var date = Utils.ddmmyyhhmmStrToDate(vars[0], vars[1]); + var dateStr = vars[0] + " " + vars[1]; if (date >= fromDate && date <= toDate) { - foreach (var fieldNo in extraFields) + foreach (var indx in flds) { - fieldList.Add(fields[fieldNo]); + if (indx >= 1000) + { + fieldList.Add(vars[indx - 1000]); + } } if (data.TryGetValue(dateStr, out var value)) @@ -12468,6 +12421,90 @@ public string GetIntervalData(string from, string to, string vars) return "[]"; } + + public string GetDailylData(string from, string to, string fields) + { + var fromDate = Utils.FromUnixTime(long.Parse(from)); + var toDate = Utils.FromUnixTime(long.Parse(to)); + var flds = (fields ?? "").Split(',').Select(int.Parse).ToArray(); + + if (flds.Length == 0) + { + return "[]"; + } + + var data = new Dictionary>(); + + // add the headers + var fldNames = new List(); + try + { + foreach(var fld in flds) + { + fldNames.Add(cumulus.DayfileFieldNames[fld]); + } + data.Add("Date", fldNames); + } + catch (Exception ex) + { + cumulus.LogErrorMessage("GetDailylData: Error processing input fields: " + ex.Message); + return "[]"; + } + + var json = new StringBuilder("[", flds.Length * 512); + + try + { + cumulus.LogDebugMessage("GetDailylData: Processing day file records"); + + if (!File.Exists(cumulus.DayFileName)) + { + cumulus.LogErrorMessage("GetDailylData: Error, dar file does not exist"); + return "[]"; + } + + cumulus.LogDebugMessage("GetDailylData: Processing day file"); + + // read the log file into a List + var lines = File.ReadAllLines(cumulus.DayFileName).ToList(); + + foreach (var line in lines) + { + var vars = line.Split(','); + var date = Utils.ddmmyyStrToDate(vars[0]); + + if (date >= fromDate && date <= toDate) + { + var fieldList = new List(); + + foreach (var indx in flds) + { + fieldList.Add(vars[indx]); + } + + data.TryAdd(vars[0], fieldList); + } + } + + foreach (var rec in data) + { + json.Append($"[\"{rec.Key}\",\"{string.Join($"\",\"", rec.Value)}\"],"); + } + + json.Length -= 1; + json.Append(']'); + + return json.ToString(); + } + catch (Exception ex) + { + cumulus.LogErrorMessage("GetDailylData: Error - " + ex.ToString()); + } + + return "[]"; + + } + public string GetCachedSqlCommands(string draw, int start, int length, string search) { try diff --git a/Updates.txt b/Updates.txt index 791e49fc..3f04ce4c 100644 --- a/Updates.txt +++ b/Updates.txt @@ -2,7 +2,8 @@ ————————————— New - Exposes the UseDataLogger setting in Station Settings > Options > Advanced -- Implements a new data viewer where you can select and view historic data from the monthly logs for a given period, for a set of data values. See: "Data logs > Interval data viewer" +- Implements a new data viewer where you can select and view historic data from the monthly logs for a given period, for a set of data values. See: "Data logs > Interval Data Viewer" +- Implements a new data viewer where you can select and view historic daily data from the day file for a given period, for a set of data values. See: "Data logs > Daily Data Viewer" Changed From b6f8489ba5c43ca3b18d8a6f8b52c7d2bcf45802 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Tue, 27 Aug 2024 21:41:35 +0100 Subject: [PATCH 13/63] Auto code cleanup --- CumulusMX/Api.cs | 18 +- CumulusMX/CalibrationsLimits.cs | 2 +- CumulusMX/Cumulus.cs | 2 +- CumulusMX/DavisAirLink.cs | 2 +- CumulusMX/DavisStation.cs | 16 +- CumulusMX/DavisWllStation.cs | 2 +- CumulusMX/EcowittApi.cs | 2 +- CumulusMX/EcowittCloudStation.cs | 2 +- CumulusMX/EcowittHttpApiStation.cs | 8 +- CumulusMX/EcowittLocalApi.cs | 7 +- CumulusMX/EmailSender.cs | 4 +- CumulusMX/GW1000Station.cs | 9 +- CumulusMX/HttpStationEcowitt.cs | 2 +- CumulusMX/ImetStation.cs | 1 - CumulusMX/JsonStation.cs | 282 ++++++++++++------------ CumulusMX/LangSettings.cs | 4 +- CumulusMX/MeteoLib.cs | 3 +- CumulusMX/NOAAReports.cs | 4 +- CumulusMX/ProgramSettings.cs | 2 +- CumulusMX/SelfInstaller.cs | 2 +- CumulusMX/StationSettings.cs | 5 - CumulusMX/ThirdParty/WebUploadWCloud.cs | 1 - CumulusMX/ThirdParty/WebUploadWow.cs | 2 +- CumulusMX/WMR100Station.cs | 2 +- CumulusMX/WS2300Station.cs | 2 +- CumulusMX/WeatherStation.cs | 12 +- CumulusMX/webtags.cs | 8 +- 27 files changed, 197 insertions(+), 209 deletions(-) diff --git a/CumulusMX/Api.cs b/CumulusMX/Api.cs index 1d92700a..c90063cf 100644 --- a/CumulusMX/Api.cs +++ b/CumulusMX/Api.cs @@ -15,7 +15,7 @@ namespace CumulusMX public static class Api { internal static WeatherStation Station { get; set; } - internal static Cumulus cumulus { get; set; } + internal static Cumulus cumulus { get; set; } public static ProgramSettings programSettings { get; set; } internal static StationSettings stationSettings { get; set; } public static InternetSettings internetSettings { get; set; } @@ -26,19 +26,19 @@ public static class Api public static MysqlSettings mySqlSettings { get; set; } public static MqttSettings mqttSettings { get; set; } public static CustomLogs customLogs { get; set; } - internal static HttpFiles httpFiles { get; set; } + internal static HttpFiles httpFiles { get; set; } public static Wizard wizard { get; set; } - internal static LangSettings langSettings { get; set; } - internal static DisplaySettings displaySettings { get; set; } - internal static AlarmSettings alarmSettings { get; set; } - internal static AlarmUserSettings alarmUserSettings { get; set; } - internal static DataEditor dataEditor { get; set; } + internal static LangSettings langSettings { get; set; } + internal static DisplaySettings displaySettings { get; set; } + internal static AlarmSettings alarmSettings { get; set; } + internal static AlarmUserSettings alarmUserSettings { get; set; } + internal static DataEditor dataEditor { get; set; } internal static ApiTagProcessor tagProcessor { get; set; } internal static HttpStationWund stationWund { get; set; } internal static HttpStationEcowitt stationEcowitt { get; set; } internal static HttpStationEcowitt stationEcowittExtra { get; set; } - internal static HttpStationAmbient stationAmbient { get; set; } - internal static HttpStationAmbient stationAmbientExtra { get; set; } + internal static HttpStationAmbient stationAmbient { get; set; } + internal static HttpStationAmbient stationAmbientExtra { get; set; } internal static JsonStation stationJson { get; set; } private static readonly char[] separator = [':']; diff --git a/CumulusMX/CalibrationsLimits.cs b/CumulusMX/CalibrationsLimits.cs index 2393ba70..46ab794e 100644 --- a/CumulusMX/CalibrationsLimits.cs +++ b/CumulusMX/CalibrationsLimits.cs @@ -62,6 +62,6 @@ public class Spikes public double PressDiff { get; set; } = 999; public double TempDiff { get; set; } = 999; public double InTempDiff { get; set; } = 999; - public double InHumDiff { get; set; }= 999; + public double InHumDiff { get; set; } = 999; } } diff --git a/CumulusMX/Cumulus.cs b/CumulusMX/Cumulus.cs index ca90bb20..d21a2f98 100644 --- a/CumulusMX/Cumulus.cs +++ b/CumulusMX/Cumulus.cs @@ -3729,7 +3729,7 @@ private void ReadIniFile() Units.Press = ini.GetValue("Station", "PressureUnit", 1, 0, 3); Units.Rain = ini.GetValue("Station", "RainUnit", 0, 0, 1); - Units.Temp = ini.GetValue("Station", "TempUnit", 0, 0 , 1); + Units.Temp = ini.GetValue("Station", "TempUnit", 0, 0, 1); StationOptions.RoundWindSpeed = ini.GetValue("Station", "RoundWindSpeed", false); StationOptions.PrimaryAqSensor = ini.GetValue("Station", "PrimaryAqSensor", -1, -1); diff --git a/CumulusMX/DavisAirLink.cs b/CumulusMX/DavisAirLink.cs index 0bd2acdc..8eae2db2 100644 --- a/CumulusMX/DavisAirLink.cs +++ b/CumulusMX/DavisAirLink.cs @@ -39,7 +39,7 @@ internal class DavisAirLink private readonly string locationStr; private readonly bool standalone; private readonly bool standaloneHistory; // Used to flag if we need to get history data on catch-up - //private DateTime airLinkLastUpdateTime + //private DateTime airLinkLastUpdateTime private readonly DiscoveredDevices discovered = new(); diff --git a/CumulusMX/DavisStation.cs b/CumulusMX/DavisStation.cs index 1e1b59dd..fe599250 100644 --- a/CumulusMX/DavisStation.cs +++ b/CumulusMX/DavisStation.cs @@ -32,7 +32,7 @@ internal class DavisStation : WeatherStation private bool stop; private int loggerInterval; - private readonly int[,] ForecastLookup = + private readonly int[,] ForecastLookup = { {14, 0, 0}, {13, 0, 0}, {12, 0, 0}, {11, 0, 0}, {13, 0, 0}, {25, 0, 0}, {24, 0, 0}, {24, 0, 0}, {10, 0, 0}, {24, 0, 0}, {24, 0, 0}, {13, 0, 0}, {7, 2, 0}, {24, 0, 0}, {13, 0, 0}, {6, 3, 0}, {13, 0, 0}, {24, 0, 0}, {13, 0, 0}, {6, 6, 0}, {13, 0, 0}, {24, 0, 0}, @@ -247,7 +247,7 @@ private void DecodeReceptionStats(string recepStats) private string GetFirmwareVersion() { cumulus.LogMessage("Reading firmware version"); - StringBuilder response = new (); + StringBuilder response = new(); StringBuilder data = new(); int ch; @@ -541,7 +541,7 @@ private string GetReceptionStats() // 0 1 23 4 5 6 cumulus.LogMessage("Reading reception stats"); lastRecepStatsTime = DateTime.Now; - StringBuilder response = new (); + StringBuilder response = new(); var bytesRead = 0; byte[] readBuffer = new byte[40]; int ch; @@ -1259,7 +1259,7 @@ private void SendBarRead() { cumulus.LogDebugMessage("Sending BARREAD"); - StringBuilder response = new (); + StringBuilder response = new(); var bytesRead = 0; byte[] readBuffer = new byte[64]; @@ -1992,7 +1992,7 @@ private void GetAndProcessLoopData(int number) private static string ProcessTxBatt(byte txStatus) { - StringBuilder response = new (); + StringBuilder response = new(); for (int i = 0; i < 8; i++) { @@ -2374,7 +2374,7 @@ private void GetArchiveData() // Read the response comport.Read(data, 0, 6); - StringBuilder resp = new ("Response:"); + StringBuilder resp = new("Response:"); for (int i = 0; i < 6; i++) { @@ -3035,7 +3035,7 @@ private bool WakeVP(TcpClient thePort, bool force = false) int retryCount = 0; // Check if we haven't sent a command within the last two minutes - use 1:50 () to be safe - if (awakeStopWatch.IsRunning && awakeStopWatch.ElapsedMilliseconds < 110000 && !force ) + if (awakeStopWatch.IsRunning && awakeStopWatch.ElapsedMilliseconds < 110000 && !force) { cumulus.LogDebugMessage("WakeVP: Not required"); awakeStopWatch.Restart(); @@ -3351,7 +3351,7 @@ private void InitTCP() try { - if (socket== null) + if (socket == null) { cumulus.LogErrorMessage("InitTCP: No TCP connection, giving up"); return; diff --git a/CumulusMX/DavisWllStation.cs b/CumulusMX/DavisWllStation.cs index 7f7c0cbd..849d93d8 100644 --- a/CumulusMX/DavisWllStation.cs +++ b/CumulusMX/DavisWllStation.cs @@ -2104,7 +2104,7 @@ private void DecodeHistoric(int dataType, int sensorType, string json) var gust = ConvertUnits.WindMPHToUser((double) data11.wind_speed_hi); var spd = ConvertUnits.WindMPHToUser((double) data11.wind_speed_avg); var dir = data11.wind_speed_hi_dir ?? 0; - var dirCal = (int)cumulus.Calib.WindDir.Calibrate(dir); + var dirCal = (int) cumulus.Calib.WindDir.Calibrate(dir); cumulus.LogDebugMessage($"WL.com historic: using wind data from TxId {data11.tx_id}"); // only record average speed values in recentwind to avoid spikes when switching to live broadcast reception DoWind(spd, dirCal, spd, recordTs); diff --git a/CumulusMX/EcowittApi.cs b/CumulusMX/EcowittApi.cs index 95a9d69f..a6e86c40 100644 --- a/CumulusMX/EcowittApi.cs +++ b/CumulusMX/EcowittApi.cs @@ -2598,7 +2598,7 @@ internal async Task GetLatestFirmwareVersion(string model, string mac, s cumulus.LogMessage("API.GetLatestFirmwareVersion: Operation throttled, retrying later..."); // delay 5 minutes and try again await Task.Delay(5 * 60 * 1000, token); - await GetLatestFirmwareVersion(model, mac,version, token); + await GetLatestFirmwareVersion(model, mac, version, token); return null; default: cumulus.LogMessage($"API.GetLatestFirmwareVersion: {retObj.msg}"); diff --git a/CumulusMX/EcowittCloudStation.cs b/CumulusMX/EcowittCloudStation.cs index 10feee70..26eb46cb 100644 --- a/CumulusMX/EcowittCloudStation.cs +++ b/CumulusMX/EcowittCloudStation.cs @@ -129,7 +129,7 @@ public EcowittCloudStation(Cumulus cumulus, WeatherStation station = null) : bas } } - _ =CheckAvailableFirmware(); + _ = CheckAvailableFirmware(); } public override void Start() diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index b8a4a5b6..62a381f6 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -8,8 +8,6 @@ using System.Threading.Tasks; using System.Timers; -using ServiceStack.Text; - namespace CumulusMX { #pragma warning disable CA1001 // Types that own disposable fields should be disposable @@ -654,8 +652,8 @@ private void GetLiveData() idx += 2; break; case 0x03: //Dew point (℃) - //tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx) - //dewpoint = tempInt16 / 10.0 + //tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx) + //dewpoint = tempInt16 / 10.0 idx += 2; break; case 0x04: //Wind chill (℃) @@ -1329,7 +1327,7 @@ private bool DoCO2DecodeNew(byte[] data, int index) idx += 2; CO2_pm4 = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; idx += 2; - CO2_pm4_24h = GW1000Api.ConvertBigEndianUInt16(data, idx)/ 10.0; + CO2_pm4_24h = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; idx += 2; var msg = $"WH45 CO₂ New: temp={CO2_temperature.ToString(cumulus.TempFormat)}, hum={CO2_humidity}, pm10={CO2_pm10:F1}, pm10_24h={CO2_pm10_24h:F1}, pm2.5={CO2_pm2p5:F1}, pm2.5_24h={CO2_pm2p5_24h:F1}, CO₂={CO2}, CO₂_24h={CO2_24h}, pm1={CO2_pm1:F1}, pm1_24h={CO2_pm1_24h:F1}, pm4={CO2_pm4:F1}, pm4_24h={CO2_pm4_24h:F1}"; var batt = TestBattery3(data[idx]); diff --git a/CumulusMX/EcowittLocalApi.cs b/CumulusMX/EcowittLocalApi.cs index 6283c6de..52628590 100644 --- a/CumulusMX/EcowittLocalApi.cs +++ b/CumulusMX/EcowittLocalApi.cs @@ -2,7 +2,6 @@ using System.Threading; using ServiceStack; -using ServiceStack.Text; @@ -232,7 +231,7 @@ private enum commonSensorTypes rainDay = 16, rainWeek = 17, rainMonth = 18, - rainYear =19, + rainYear = 19, rainTotals = 20, light = 21, uv = 22, @@ -243,8 +242,8 @@ private enum commonSensorTypes public class commonSensor { - public int id { get; set ; } - public string val { get; set; } + public int id { get; set; } + public string val { get; set; } public string? unit { get; set; } public double? battery { get; set; } diff --git a/CumulusMX/EmailSender.cs b/CumulusMX/EmailSender.cs index d35fd2dc..13e3474c 100644 --- a/CumulusMX/EmailSender.cs +++ b/CumulusMX/EmailSender.cs @@ -17,7 +17,7 @@ namespace CumulusMX public class EmailSender(Cumulus cumulus) { static readonly Regex ValidEmailRegex = CreateValidEmailRegex(); - private static readonly SemaphoreSlim _writeLock = new (1); + private static readonly SemaphoreSlim _writeLock = new(1); private readonly Cumulus cumulus = cumulus; public async Task SendEmail(string[] to, string from, string subject, string message, bool isHTML, bool useBcc) @@ -234,7 +234,7 @@ public static bool CheckEmailAddress(string email) public class SmtpOptions { - public bool Enabled { get; set; } + public bool Enabled { get; set; } public string Server { get; set; } public int Port { get; set; } public string User { get; set; } diff --git a/CumulusMX/GW1000Station.cs b/CumulusMX/GW1000Station.cs index 7d453761..212dd2a4 100644 --- a/CumulusMX/GW1000Station.cs +++ b/CumulusMX/GW1000Station.cs @@ -730,7 +730,8 @@ private bool PrintSensorInfoNew(byte[] data, int idx) try { type = Enum.GetName(typeof(GW1000Api.SensorIds), data[idx]).ToUpper(); - } catch + } + catch { // leave the device id as null } @@ -946,8 +947,8 @@ private void GetLiveData() idx += 2; break; case 0x03: //Dew point (℃) - //tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx) - //dewpoint = tempInt16 / 10.0 + //tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx) + //dewpoint = tempInt16 / 10.0 idx += 2; break; case 0x04: //Wind chill (℃) @@ -1735,7 +1736,7 @@ private bool DoCO2DecodeNew(byte[] data, int index) idx += 2; CO2_pm4 = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; idx += 2; - CO2_pm4_24h = GW1000Api.ConvertBigEndianUInt16(data, idx)/ 10.0; + CO2_pm4_24h = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; idx += 2; var msg = $"WH45 CO₂ New: temp={CO2_temperature.ToString(cumulus.TempFormat)}, hum={CO2_humidity}, pm10={CO2_pm10:F1}, pm10_24h={CO2_pm10_24h:F1}, pm2.5={CO2_pm2p5:F1}, pm2.5_24h={CO2_pm2p5_24h:F1}, CO₂={CO2}, CO₂_24h={CO2_24h}, pm1={CO2_pm1:F1}, pm1_24h={CO2_pm1_24h:F1}, pm4={CO2_pm4:F1}, pm4_24h={CO2_pm4_24h:F1}"; var batt = TestBattery3(data[idx]); diff --git a/CumulusMX/HttpStationEcowitt.cs b/CumulusMX/HttpStationEcowitt.cs index 9128148b..b3899ca7 100644 --- a/CumulusMX/HttpStationEcowitt.cs +++ b/CumulusMX/HttpStationEcowitt.cs @@ -170,7 +170,7 @@ public HttpStationEcowitt(Cumulus cumulus, WeatherStation station = null) : base } else { - _= CheckAvailableFirmware(deviceModel); + _ = CheckAvailableFirmware(deviceModel); } } diff --git a/CumulusMX/ImetStation.cs b/CumulusMX/ImetStation.cs index 2dbbc6c6..3c8603e6 100644 --- a/CumulusMX/ImetStation.cs +++ b/CumulusMX/ImetStation.cs @@ -3,7 +3,6 @@ using System.ComponentModel; using System.Globalization; using System.IO.Ports; -using System.Text; using System.Threading; namespace CumulusMX diff --git a/CumulusMX/JsonStation.cs b/CumulusMX/JsonStation.cs index a312eb55..5689646b 100644 --- a/CumulusMX/JsonStation.cs +++ b/CumulusMX/JsonStation.cs @@ -269,19 +269,19 @@ private string ApplyData(string dataString) // Humidity try { - if (data.humidity != null) - { - if (data.humidity.outdoor != null) - { - DoOutdoorHumidity(data.humidity.outdoor.Value, data.lastupdated); - haveHum = true; - } - if (data.humidity.indoor != null) + if (data.humidity != null) { - DoIndoorHumidity(data.humidity.indoor.Value); + if (data.humidity.outdoor != null) + { + DoOutdoorHumidity(data.humidity.outdoor.Value, data.lastupdated); + haveHum = true; + } + if (data.humidity.indoor != null) + { + DoIndoorHumidity(data.humidity.indoor.Value); + } } } - } catch (Exception ex) { cumulus.LogExceptionMessage(ex, "ApplyData: Error processing humidity"); @@ -292,60 +292,60 @@ private string ApplyData(string dataString) // Wind try { - if (data.wind != null && data.units != null) - { - if (data.units.windspeed == null) + if (data.wind != null && data.units != null) { - cumulus.LogErrorMessage("ApplyData: No windspeed units supplied!"); - retStr.AppendLine("No windspeed units"); - } - else - { - var avg = data.wind.speed ?? -1; - var gust = data.wind.gust10m ?? -1; - - if (gust < 0) + if (data.units.windspeed == null) { - cumulus.LogErrorMessage("ApplyData: No gust value supplied in wind data"); - retStr.AppendLine("No gust value supplied in wind data"); + cumulus.LogErrorMessage("ApplyData: No windspeed units supplied!"); + retStr.AppendLine("No windspeed units"); } else { - var doit = true; - switch (data.units.windspeed) + var avg = data.wind.speed ?? -1; + var gust = data.wind.gust10m ?? -1; + + if (gust < 0) + { + cumulus.LogErrorMessage("ApplyData: No gust value supplied in wind data"); + retStr.AppendLine("No gust value supplied in wind data"); + } + else { - case "mph": - avg = ConvertUnits.WindMPHToUser(avg); - gust = ConvertUnits.WindMPHToUser(gust); - break; - case "ms": - avg = ConvertUnits.WindMSToUser(avg); - gust = ConvertUnits.WindMSToUser(gust); - break; - case "kph": - avg = ConvertUnits.WindKPHToUser(avg); - gust = ConvertUnits.WindKPHToUser(gust); - break; - case "knots": + var doit = true; + switch (data.units.windspeed) + { + case "mph": + avg = ConvertUnits.WindMPHToUser(avg); + gust = ConvertUnits.WindMPHToUser(gust); + break; + case "ms": + avg = ConvertUnits.WindMSToUser(avg); + gust = ConvertUnits.WindMSToUser(gust); + break; + case "kph": + avg = ConvertUnits.WindKPHToUser(avg); + gust = ConvertUnits.WindKPHToUser(gust); + break; + case "knots": avg = ConvertUnits.WindKnotsToUser(avg); gust = ConvertUnits.WindKnotsToUser(gust); break; - default: - cumulus.LogErrorMessage("ApplyData: Invalid windspeed units supplied: " + data.units.windspeed); - retStr.AppendLine("Invalid windspeed units"); - doit = false; - break; - } + default: + cumulus.LogErrorMessage("ApplyData: Invalid windspeed units supplied: " + data.units.windspeed); + retStr.AppendLine("Invalid windspeed units"); + doit = false; + break; + } - if (doit) - { - DoWind(gust, data.wind.direction ?? 0, avg, data.lastupdated); - haveWind = true; + if (doit) + { + DoWind(gust, data.wind.direction ?? 0, avg, data.lastupdated); + haveWind = true; + } } } } } - } catch (Exception ex) { cumulus.LogExceptionMessage(ex, "ApplyData: Error processing wind"); @@ -356,58 +356,58 @@ private string ApplyData(string dataString) // Rain try { - if (data.rain != null && data.units != null) - { - var doit = true; - double counter = 0; - if (data.rain.counter.HasValue) - { - counter = data.rain.counter.Value; - } - else if (data.rain.year.HasValue) - { - counter = data.rain.year.Value; - } - else + if (data.rain != null && data.units != null) { - cumulus.LogErrorMessage("ApplyData: No rainfall counter/year value supplied!"); - retStr.AppendLine("No rainfall counter/year value supplied"); - doit = false; - } - - if (doit) - { - if (data.units.rainfall == null) + var doit = true; + double counter = 0; + if (data.rain.counter.HasValue) { - cumulus.LogErrorMessage("ApplyData: No rainfall units supplied!"); - retStr.AppendLine("No rainfall units"); + counter = data.rain.counter.Value; } - else if (data.units.rainfall == "mm") + else if (data.rain.year.HasValue) { - var rate = ConvertUnits.RainMMToUser(data.rain.rate ?? 0); + counter = data.rain.year.Value; + } + else + { + cumulus.LogErrorMessage("ApplyData: No rainfall counter/year value supplied!"); + retStr.AppendLine("No rainfall counter/year value supplied"); + doit = false; + } - if (data.rain.counter.HasValue) + if (doit) + { + if (data.units.rainfall == null) { - DoRain(ConvertUnits.RainMMToUser(counter), rate, data.lastupdated); + cumulus.LogErrorMessage("ApplyData: No rainfall units supplied!"); + retStr.AppendLine("No rainfall units"); } - } - else if (data.units.rainfall == "in") - { - var rate = ConvertUnits.RainINToUser(data.rain.rate ?? 0); + else if (data.units.rainfall == "mm") + { + var rate = ConvertUnits.RainMMToUser(data.rain.rate ?? 0); + + if (data.rain.counter.HasValue) + { + DoRain(ConvertUnits.RainMMToUser(counter), rate, data.lastupdated); + } + } + else if (data.units.rainfall == "in") + { + var rate = ConvertUnits.RainINToUser(data.rain.rate ?? 0); - if (data.rain.counter.HasValue) + if (data.rain.counter.HasValue) + { + DoRain(ConvertUnits.RainINToUser(counter), rate, data.lastupdated); + } + } + else { - DoRain(ConvertUnits.RainINToUser(counter), rate, data.lastupdated); + cumulus.LogErrorMessage("ApplyData: Invalid rainfall units supplied = " + data.units.rainfall); + retStr.AppendLine($"Invalid rainfall units: {data.units.rainfall}"); } } - else - { - cumulus.LogErrorMessage("ApplyData: Invalid rainfall units supplied = " + data.units.rainfall); - retStr.AppendLine($"Invalid rainfall units: {data.units.rainfall}"); - } } } - } catch (Exception ex) { cumulus.LogExceptionMessage(ex, "ApplyData: Error processing rain"); @@ -418,72 +418,72 @@ private string ApplyData(string dataString) // Pressure try { - if (data.pressure != null && data.units != null) - { - if (data.units.pressure == null) + if (data.pressure != null && data.units != null) { - cumulus.LogErrorMessage("ApplyData: No pressure units supplied!"); - retStr.AppendLine("No pressure units"); - } - else - { - var slp = data.pressure.sealevel ?? -1; - var abs = data.pressure.absolute ?? -1; - - if (slp < 0 && abs < 0) + if (data.units.pressure == null) { - cumulus.LogErrorMessage("ApplyData: No pressure values in data"); - retStr.AppendLine("No pressure values in data"); + cumulus.LogErrorMessage("ApplyData: No pressure units supplied!"); + retStr.AppendLine("No pressure units"); } else { - var doit = true; + var slp = data.pressure.sealevel ?? -1; + var abs = data.pressure.absolute ?? -1; - if (cumulus.StationOptions.CalculateSLP && abs < 0) + if (slp < 0 && abs < 0) { - cumulus.LogErrorMessage("ApplyData: Calculate SLP is enabled, but no abosolute pressure value in data"); - retStr.AppendLine("Calculate SLP is enabled, but no abosolute pressure value in data"); + cumulus.LogErrorMessage("ApplyData: No pressure values in data"); + retStr.AppendLine("No pressure values in data"); } else { - switch (data.units.pressure) + var doit = true; + + if (cumulus.StationOptions.CalculateSLP && abs < 0) { - case "hPa": - slp = ConvertUnits.PressMBToUser(slp); - StationPressure = ConvertUnits.PressMBToUser(abs); - break; - case "kPa": - slp = ConvertUnits.PressKPAToUser(slp); - StationPressure = ConvertUnits.PressKPAToUser(abs); - break; - case "inHg": - slp = ConvertUnits.PressINHGToUser(slp); - StationPressure = ConvertUnits.PressINHGToUser(abs); - break; - default: - cumulus.LogErrorMessage("ApplyData: Invalid pressure units supplied: " + data.units.pressure); - retStr.AppendLine("Invalid pressure units"); - doit = false; - break; + cumulus.LogErrorMessage("ApplyData: Calculate SLP is enabled, but no abosolute pressure value in data"); + retStr.AppendLine("Calculate SLP is enabled, but no abosolute pressure value in data"); } - - StationPressure = cumulus.Calib.Press.Calibrate(StationPressure); - - if (doit) + else { - if (cumulus.StationOptions.CalculateSLP) + switch (data.units.pressure) { - slp = MeteoLib.GetSeaLevelPressure(cumulus.Altitude, ConvertUnits.UserPressToHpa(StationPressure), OutdoorTemperature); - slp = ConvertUnits.PressMBToUser(slp); + case "hPa": + slp = ConvertUnits.PressMBToUser(slp); + StationPressure = ConvertUnits.PressMBToUser(abs); + break; + case "kPa": + slp = ConvertUnits.PressKPAToUser(slp); + StationPressure = ConvertUnits.PressKPAToUser(abs); + break; + case "inHg": + slp = ConvertUnits.PressINHGToUser(slp); + StationPressure = ConvertUnits.PressINHGToUser(abs); + break; + default: + cumulus.LogErrorMessage("ApplyData: Invalid pressure units supplied: " + data.units.pressure); + retStr.AppendLine("Invalid pressure units"); + doit = false; + break; } - DoPressure(slp, data.lastupdated); + StationPressure = cumulus.Calib.Press.Calibrate(StationPressure); + + if (doit) + { + if (cumulus.StationOptions.CalculateSLP) + { + slp = MeteoLib.GetSeaLevelPressure(cumulus.Altitude, ConvertUnits.UserPressToHpa(StationPressure), OutdoorTemperature); + slp = ConvertUnits.PressMBToUser(slp); + } + + DoPressure(slp, data.lastupdated); + } } } } } } - } catch (Exception ex) { cumulus.LogExceptionMessage(ex, "ApplyData: Error processing pressure"); @@ -759,7 +759,7 @@ private string ApplyData(string dataString) } -#pragma warning disable S3459,S1144 // Unused private types or members should be removed +#pragma warning disable S3459, S1144 // Unused private types or members should be removed private sealed class DataObject { public UnitsObject units { get; set; } @@ -785,13 +785,13 @@ private sealed class UnitsObject public string windspeed { get; set; } public string rainfall { get; set; } public string pressure { get; set; } - public string soilmoisture { get; set;} + public string soilmoisture { get; set; } } private sealed class Temperature { public double? outdoor { get; set; } - public double? indoor { get; set;} + public double? indoor { get; set; } public double? dewpoint { get; set; } } private sealed class Humidity @@ -803,7 +803,7 @@ private sealed class Wind { public double? speed { get; set; } public int? direction { get; set; } - public double? gust10m { get; set;} + public double? gust10m { get; set; } } private sealed class Rain { @@ -813,13 +813,13 @@ private sealed class Rain } private sealed class PressureJson { - public double? absolute { get; set;} + public double? absolute { get; set; } public double? sealevel { get; set; } } private sealed class Solar { public int? irradiation { get; set; } - public double? uvi { get; set;} + public double? uvi { get; set; } } private class ExtraTempJson { @@ -841,7 +841,7 @@ private class PmData public double? pm2p5 { get; set; } public double? pm2p5avg24h { get; set; } public double? pm10 { get; set; } - public double? pm10avg24h { get; set;} + public double? pm10avg24h { get; set; } } private sealed class Co2Data : PmData { diff --git a/CumulusMX/LangSettings.cs b/CumulusMX/LangSettings.cs index b7261505..48868d7a 100644 --- a/CumulusMX/LangSettings.cs +++ b/CumulusMX/LangSettings.cs @@ -469,8 +469,8 @@ public string UpdateConfig(IHttpContext context) } - // Save the settings - cumulus.WriteStringsFile(); + // Save the settings + cumulus.WriteStringsFile(); } catch (Exception ex) { diff --git a/CumulusMX/MeteoLib.cs b/CumulusMX/MeteoLib.cs index 4f643f42..2f53d808 100644 --- a/CumulusMX/MeteoLib.cs +++ b/CumulusMX/MeteoLib.cs @@ -1,5 +1,4 @@ using System; -using System.CodeDom; namespace CumulusMX { @@ -47,7 +46,7 @@ public static double WindChill(double tempC, double windSpeedKph, bool tempCutof public static double ApparentTemperature(double tempC, double windspeedMS, int humidity) { double avp = (humidity / 100.0) * 6.105 * Math.Exp(17.27 * tempC / (237.7 + tempC)); // hPa - //double avp = ActualVapourPressure(tempC, humidity) + //double avp = ActualVapourPressure(tempC, humidity) return tempC + (0.33 * avp) - (0.7 * windspeedMS) - 4.0; } diff --git a/CumulusMX/NOAAReports.cs b/CumulusMX/NOAAReports.cs index a6e9f1bf..6f53a97f 100644 --- a/CumulusMX/NOAAReports.cs +++ b/CumulusMX/NOAAReports.cs @@ -4,8 +4,6 @@ using System.Text; using System.Threading.Tasks; -using HidSharp.Reports; - namespace CumulusMX { internal class NOAAReports(Cumulus cumulus, WeatherStation station) @@ -268,7 +266,7 @@ public string GetLastNoaaMonthReportFilename(DateTime dat, bool fullPath) return logfiledate.AddHours(-1).ToString(cumulus.NOAAconf.MonthFile); } - public string UploadNoaaReport(int year, int month=-1) + public string UploadNoaaReport(int year, int month = -1) { DateTime noaats = month == -1 ? new DateTime(year, 1, 1) : new DateTime(year, month, 1); var reportName = string.Empty; diff --git a/CumulusMX/ProgramSettings.cs b/CumulusMX/ProgramSettings.cs index b1505518..876b1f1e 100644 --- a/CumulusMX/ProgramSettings.cs +++ b/CumulusMX/ProgramSettings.cs @@ -178,7 +178,7 @@ public string UpdateConfig(IHttpContext context) cumulus.SetupFtpLogging(cumulus.FtpOptions.Logging); cumulus.SetRealTimeFtpLogging(cumulus.FtpOptions.Logging); } - else if (settings.logging.ftplogginglevel.HasValue && cumulus.FtpOptions.LoggingLevel!= settings.logging.ftplogginglevel.Value) + else if (settings.logging.ftplogginglevel.HasValue && cumulus.FtpOptions.LoggingLevel != settings.logging.ftplogginglevel.Value) { cumulus.FtpOptions.LoggingLevel = settings.logging.ftplogginglevel.Value; cumulus.SetupFtpLogging(cumulus.FtpOptions.Logging); diff --git a/CumulusMX/SelfInstaller.cs b/CumulusMX/SelfInstaller.cs index 547a9de7..b85146ef 100644 --- a/CumulusMX/SelfInstaller.cs +++ b/CumulusMX/SelfInstaller.cs @@ -196,7 +196,7 @@ private static int RunCommand(string exe, string args, bool createWindow = false Arguments = args }; - if (elevated ) + if (elevated) { startinfo.Verb = "runas"; } diff --git a/CumulusMX/StationSettings.cs b/CumulusMX/StationSettings.cs index 31426315..1342e4c0 100644 --- a/CumulusMX/StationSettings.cs +++ b/CumulusMX/StationSettings.cs @@ -8,14 +8,9 @@ using EmbedIO; -using MQTTnet.Protocol; -using MQTTnet.Server; - using ServiceStack; using ServiceStack.Text; -using static SQLite.SQLite3; - namespace CumulusMX { diff --git a/CumulusMX/ThirdParty/WebUploadWCloud.cs b/CumulusMX/ThirdParty/WebUploadWCloud.cs index 549a4d24..3a9afce5 100644 --- a/CumulusMX/ThirdParty/WebUploadWCloud.cs +++ b/CumulusMX/ThirdParty/WebUploadWCloud.cs @@ -1,5 +1,4 @@ using System; -using System.Net.Http; using System.Text; using System.Threading.Tasks; diff --git a/CumulusMX/ThirdParty/WebUploadWow.cs b/CumulusMX/ThirdParty/WebUploadWow.cs index 1f7d4f85..f43f4697 100644 --- a/CumulusMX/ThirdParty/WebUploadWow.cs +++ b/CumulusMX/ThirdParty/WebUploadWow.cs @@ -41,7 +41,7 @@ internal override async Task DoUpdate(DateTime timestamp) var maxRetryAttempts = 2; var delay = maxRetryAttempts * 5.0; - for (int retryCount = maxRetryAttempts; retryCount >= 0 ; retryCount--) + for (int retryCount = maxRetryAttempts; retryCount >= 0; retryCount--) { try { diff --git a/CumulusMX/WMR100Station.cs b/CumulusMX/WMR100Station.cs index bb6df66f..4f5b6eeb 100644 --- a/CumulusMX/WMR100Station.cs +++ b/CumulusMX/WMR100Station.cs @@ -232,7 +232,7 @@ private void RemovePacketFromBuffer() private void ProcessWMR100Packet() { - StringBuilder str =new (); + StringBuilder str = new(); for (int i = 0; i <= currentPacketLength - 3; i++) { diff --git a/CumulusMX/WS2300Station.cs b/CumulusMX/WS2300Station.cs index 19c8ef8e..b53e9d4e 100644 --- a/CumulusMX/WS2300Station.cs +++ b/CumulusMX/WS2300Station.cs @@ -693,7 +693,7 @@ private void GetAndProcessData() { DoWindChill(ConvertUnits.TempCToUser(wc), now); } - } + } // Rain =========================================================================== if (!stop) diff --git a/CumulusMX/WeatherStation.cs b/CumulusMX/WeatherStation.cs index 7238dd6e..0f89fcc4 100644 --- a/CumulusMX/WeatherStation.cs +++ b/CumulusMX/WeatherStation.cs @@ -2115,7 +2115,7 @@ private void CheckForDataStopped() { cumulus.LogErrorMessage("*** Data input appears to have stopped"); } - } // Calculates evapotranspiration based on the data for the last hour and updates the running annual total. + } // Calculates evapotranspiration based on the data for the last hour and updates the running annual total. else { DataStopped = false; @@ -2183,7 +2183,7 @@ public void CreateGraphDataFiles() // Chart data for Highcharts graphs string json; // 0=graphconfig, 1=availabledata, 8=dailyrain, 9=dailytemp, 11=sunhours - int[] createReqOnce = [0,1,8,9,11]; + int[] createReqOnce = [0, 1, 8, 9, 11]; for (var i = 0; i < cumulus.GraphDataFiles.Length; i++) { @@ -6695,7 +6695,7 @@ private static int getShortestAngle(int bearing1, int bearing2) public readonly double[] WindRunHourMult = [3.6, 1.0, 1.0, 1.0]; public DateTime LastDataReadTimestamp = DateTime.MinValue; // Stored in UTC to avoid clock change issues public DateTime SavedLastDataReadTimestamp = DateTime.MinValue; // Stored in UTC to avoid clock change issues - // Create arrays with 9 entries, 0 = VP2, 1-8 = WLL TxIds + // Create arrays with 9 entries, 0 = VP2, 1-8 = WLL TxIds public int DavisTotalPacketsReceived = 0; public int[] DavisTotalPacketsMissed = [0, 0, 0, 0, 0, 0, 0, 0, 0]; public int[] DavisNumberOfResynchs = [0, 0, 0, 0, 0, 0, 0, 0, 0]; @@ -8910,7 +8910,7 @@ public string LoadDayFile() { int addedEntries = 0; - StringBuilder msg = new (); + StringBuilder msg = new(); cumulus.LogMessage("LoadDayFile: Attempting to load the day file"); if (dayfileReloading) @@ -9547,7 +9547,7 @@ public double GetAqi(AqMeasure type, double value) }; case 4: // Canada AQHI - // return AirQualityIndices.CA_AQHI(value) + // return AirQualityIndices.CA_AQHI(value) return -1; case 5: // Australia NEPM @@ -12439,7 +12439,7 @@ public string GetDailylData(string from, string to, string fields) var fldNames = new List(); try { - foreach(var fld in flds) + foreach (var fld in flds) { fldNames.Add(cumulus.DayfileFieldNames[fld]); } diff --git a/CumulusMX/webtags.cs b/CumulusMX/webtags.cs index 77ce32d3..5dfea56d 100644 --- a/CumulusMX/webtags.cs +++ b/CumulusMX/webtags.cs @@ -3551,7 +3551,7 @@ private string TagSunshineHoursMonth(Dictionary tagParams) else { end = DateTime.Now; - start = new DateTime(end.Year, end.Month, 1, 0 ,0, 0, DateTimeKind.Local); + start = new DateTime(end.Year, end.Month, 1, 0, 0, 0, DateTimeKind.Local); } return CheckRcDp(station.DayFile.Where(rec => rec.Date >= start && rec.Date < end).Sum(rec => rec.SunShineHours == Cumulus.DefaultHiVal ? 0 : rec.SunShineHours), tagParams, 1); @@ -3593,7 +3593,7 @@ private string TagMonthTempAvg(Dictionary tagParams) if (year != null && month != null) { - start = new DateTime(int.Parse(year), int.Parse(month), 1, 0 ,0, 0, DateTimeKind.Local); + start = new DateTime(int.Parse(year), int.Parse(month), 1, 0, 0, 0, DateTimeKind.Local); end = start.AddMonths(1); } else @@ -5352,7 +5352,7 @@ private static string TagOsVersion(Dictionary tagParams) } catch { - return Environment.OSVersion.ToString(); + return Environment.OSVersion.ToString(); } } @@ -6190,7 +6190,7 @@ public void InitialiseWebtags() { "graphperiod", Taggraphperiod }, { "stationtype", Tagstationtype }, { "stationtypeJsEnc", TagstationtypeJsEnc }, - { "stationId", TagstationId }, + { "stationId", TagstationId }, { "latitude", Taglatitude }, { "latitudeJsEnc", TaglatitudeJsEnc }, { "longitude", Taglongitude }, From 8b4f595f5dbad4723f0a2a910bb845c0e437588f Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Fri, 30 Aug 2024 12:22:50 +0100 Subject: [PATCH 14/63] SQLite update --- CumulusMX/CumulusMX.csproj | 2 +- Updates.txt | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/CumulusMX/CumulusMX.csproj b/CumulusMX/CumulusMX.csproj index b57fde54..42b1a457 100644 --- a/CumulusMX/CumulusMX.csproj +++ b/CumulusMX/CumulusMX.csproj @@ -90,7 +90,7 @@ - + diff --git a/Updates.txt b/Updates.txt index 9092493c..1191e443 100644 --- a/Updates.txt +++ b/Updates.txt @@ -8,7 +8,7 @@ Changed Package Updates -- SQLite: Reverted to v2.1.8 pending fix from author +- SQLite From 56e1ddf3b9359c1061ff77786144f5b82835491f Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sat, 31 Aug 2024 21:52:43 +0100 Subject: [PATCH 15/63] A semi-functional Ecowitt HTTP Local API station! --- CumulusMX/Cumulus.cs | 5 +- CumulusMX/CumulusMX.csproj | 2 +- CumulusMX/DavisCloudStation.cs | 7 +- CumulusMX/EcowittHttpApiStation.cs | 1517 +++++++++++----------------- CumulusMX/EcowittLocalApi.cs | 70 +- CumulusMX/StationSettings.cs | 27 + CumulusMX/Wizard.cs | 6 + Updates.txt | 3 +- 8 files changed, 677 insertions(+), 960 deletions(-) diff --git a/CumulusMX/Cumulus.cs b/CumulusMX/Cumulus.cs index d21a2f98..0c70c69d 100644 --- a/CumulusMX/Cumulus.cs +++ b/CumulusMX/Cumulus.cs @@ -491,10 +491,11 @@ public struct MqttConfig "Ecowitt Cloud", // 18 "Davis Cloud (WLL/WLC)", // 19 "Davis Cloud (VP2)", // 20 - "JSON Data" // 21 + "JSON Data", // 21 + "Ecowitt HTTP API" // 22 ]; - internal string[] APRSstationtype = ["DsVP", "DsVP", "WMR928", "WM918", "EW", "FO", "WS2300", "FOs", "WMR100", "WMR200", "IMET", "DsVP", "Ecow", "Unkn", "Ecow", "Ambt", "Tmpt", "Simul", "Ecow", "DsVP", "DsVP", "Json"]; + internal string[] APRSstationtype = ["DsVP", "DsVP", "WMR928", "WM918", "EW", "FO", "WS2300", "FOs", "WMR100", "WMR200", "IMET", "DsVP", "Ecow", "Unkn", "Ecow", "Ambt", "Tmpt", "Simul", "Ecow", "DsVP", "DsVP", "Json", "Ecow"]; internal string[] DayfileFieldNames = ["Date", "HighWindGust", "HighGustBearing", "HighGustTime", "MinTemperature", "MinTempTime", "MaxTemperature", "MaxTempTime", "MinPressure", "MinPressureTime", "MaxPressure", "MaxPressureTime", "MaxRainfallRate", "MaxRainRateTime", "TotalRainfallToday", "AvgTemperatureToday", "TotalWindRun", "HighAverageWindSpeed", "HighAvgWindSpeedTime", "LowHumidity", "LowHumidityTime", "HighHumidity", "HighHumidityTime", "TotalEvapotranspiration", "TotalHoursOfSunshine", "HighHeatIndex", "HighHeatIndexTime", "HighApparentTemperature", "HighAppTempTime", "LowApparentTemperature", "LowAppTempTime", "High1hRain", "High1hRainTime", "LowWindChill", "LowWindChillTime", "HighDewPoint", "HighDewPointTime", "LowDewPoint", "LowDewPointTime", "DominantWindBearing", "HeatingDegreeDays", "CoolingDegreeDays", "HighSolarRad", "HighSolarRadTime", "HighUv-I", "HighUv-ITime", "HighFeelsLike", "HighFeelsLikeTime", "LowFeelsLike", "LowFeelsLikeTime", "HighHumidex", "HighHumidexTime", "ChillHours", "High24hRain", "High24hRainTime"]; internal string[] LogFileFieldNames = ["Date", "Time", "Temperature", "Humidity", "DewPoint", "WindSpeed", "RecentHighGust", "AverageWindBearing", "RainfallRate", "RainfallSoFar", "SeaLevelPressure", "RainfallCounter", "InsideTemperature", "InsideHumidity", "CurrentGust", "WindChill", "HeatIndex", "UvIndex", "SolarRadiation", "Evapotranspiration", "AnnualEvapotranspiration", "ApparentTemperature", "MaxSolarRadiation", "HoursOfSunshine", "WindBearing", "Rg-11Rain", "RainSinceMidnight", "FeelsLike", "Humidex"]; diff --git a/CumulusMX/CumulusMX.csproj b/CumulusMX/CumulusMX.csproj index 6a70abb0..2ad57592 100644 --- a/CumulusMX/CumulusMX.csproj +++ b/CumulusMX/CumulusMX.csproj @@ -55,7 +55,7 @@ - 4.2.0.4030 + 4.2.0.4031 Copyright © 2015-$([System.DateTime]::Now.ToString('yyyy')) Cumulus MX diff --git a/CumulusMX/DavisCloudStation.cs b/CumulusMX/DavisCloudStation.cs index 209066da..a9f03485 100644 --- a/CumulusMX/DavisCloudStation.cs +++ b/CumulusMX/DavisCloudStation.cs @@ -2092,7 +2092,8 @@ private void DecodeCurrent(List sensors) { if (StationPressure > 0) { - var slp = MeteoLib.GetSeaLevelPressure(AltitudeM(cumulus.Altitude), ConvertUnits.UserPressToHpa(StationPressure), ConvertUnits.UserTempToC(OutdoorTemperature), cumulus.Latitude); + var stnCal = cumulus.Calib.Press.Calibrate(StationPressure); + var slp = MeteoLib.GetSeaLevelPressure(AltitudeM(cumulus.Altitude), ConvertUnits.UserPressToHpa(stnCal), ConvertUnits.UserTempToC(OutdoorTemperature), cumulus.Latitude); DoPressure(ConvertUnits.PressMBToUser(slp), dateTime); } else @@ -3312,8 +3313,8 @@ private void DecodeHistoric(int dataType, int sensorType, string json, bool curr // Altimeter from absolute if (data13baro.bar_absolute != null) { - var abs = ConvertUnits.PressINHGToHpa((double) data13baro.bar_absolute); - abs = cumulus.StationOptions.CalculateSLP ? cumulus.Calib.Press.Calibrate(abs) : abs; + // leave possible calculation of SLP until later when we have temp and humidity + StationPressure = ConvertUnits.PressINHGToUser((double) data13baro.bar_absolute); // Or do we use calibration? The VP2 code doesn't? AltimeterPressure = ConvertUnits.PressMBToUser(MeteoLib.StationToAltimeter(ConvertUnits.UserPressToHpa(StationPressure), AltitudeM(cumulus.Altitude))); diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index 62a381f6..2a4939da 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -1,13 +1,14 @@ using System; using System.Collections.Generic; using System.Linq; -using System.Net; -using System.Net.Sockets; -using System.Text; using System.Threading; using System.Threading.Tasks; using System.Timers; +using static System.Runtime.InteropServices.JavaScript.JSType; +using static CumulusMX.GW1000Api; + + namespace CumulusMX { #pragma warning disable CA1001 // Types that own disposable fields should be disposable @@ -38,6 +39,19 @@ internal class EcowittHttpApiStation : WeatherStation internal static readonly char[] dotSeparator = ['.']; internal static readonly string[] underscoreV = ["_V"]; + // local variables to hold data until all sensors have been read. Then they are set and derviced values calculated + double windSpeedLast = -999, rainRateLast = -999, rainLast = -999, gustLast = -999; + int windDirLast = -999; + double outdoortemp = -999; + double dewpoint; + double windchill = -999; + bool batteryLow = false; + + // We check the new value against what we have already, if older then ignore it! + double newLightningDistance = 999; + DateTime newLightningTime = new DateTime(1900, 1, 1, 0, 0, 0, 0, DateTimeKind.Utc); + + public EcowittHttpApiStation(Cumulus cumulus) : base(cumulus) { cumulus.Units.AirQualityUnitText = "µg/m³"; @@ -104,8 +118,6 @@ public EcowittHttpApiStation(Cumulus cumulus) : base(cumulus) ecowittApi = new EcowittApi(cumulus, this); - DoDiscovery(); - _ = CheckAvailableFirmware(); LoadLastHoursFromDataLogs(cumulus.LastUpdateTime); @@ -137,7 +149,6 @@ public override void Start() { try { - var piezoLastRead = DateTime.MinValue; var dataLastRead = DateTime.MinValue; double delay; @@ -146,23 +157,135 @@ public override void Start() var rawData = localApi.GetLiveData(cumulus.cancellationToken); dataLastRead = DateTime.Now; - // every 30 seconds read the rain rate - if ((cumulus.Gw1000PrimaryRainSensor == 1 || cumulus.StationOptions.UseRainForIsRaining == 2) && (DateTime.UtcNow - piezoLastRead).TotalSeconds >= 30 && !cumulus.cancellationToken.IsCancellationRequested) + // process the common_list sensors + ProcessCommonList(rawData.common_list, dataLastRead); + + // process base station values + ProcessWh25(rawData.wh25, dataLastRead); + + // process rain values + if (cumulus.Gw1000PrimaryRainSensor == 0 && rawData.rain != null) + { + ProcessRain(rawData.rain, dataLastRead); + // TODO: battery status for piezo + } + else if (cumulus.Gw1000PrimaryRainSensor == 1 && rawData.piezoRain != null) { - GetPiezoRainData(); - piezoLastRead = DateTime.UtcNow; + ProcessRain(rawData.piezoRain, dataLastRead); + // TODO: battery status for tipper } - var minute = DateTime.Now.Minute; - if (minute != lastMinute) + if (rawData.lightning != null) { - lastMinute = minute; + ProcessLightning(rawData.lightning, dataLastRead); + } + + if (rawData.co2 != null) + { + ProcessCo2(rawData.co2, dataLastRead); + } + + if (rawData.ch_pm25 != null) + { + ProcessChPm25(rawData.ch_pm25); + } + + if (rawData.ch_leak != null) + { + ProcessLeak(rawData.ch_leak); + } + + if (rawData.ch_aisle != null) + { + ProcessExtraTempHum(rawData.ch_aisle, dataLastRead); + } + + if (rawData.ch_temp != null) + { + ProcessUserTemp(rawData.ch_temp, dataLastRead); + } + + if (rawData.ch_soil != null) + { + ProcessSoilMoisture(rawData.ch_soil); + } + + // TODO: Soil Temperature sensors + //if (rawData.ch_ ??? != null) + //{ + // ProcessSoilTemp(rawData.ch_ ???); + //} + + if (rawData.ch_leaf != null) + { + ProcessLeafWet(rawData.ch_leaf); + } + + // Now do the stuff that requires more than one input parameter + + // Only set the lightning time/distance if it is newer than what we already have - the GW1000 seems to reset this value + if (newLightningTime > LightningTime) + { + LightningTime = newLightningTime; + if (newLightningDistance < 999) + LightningDistance = newLightningDistance; + } + + // Process outdoor temperature here, as GW1000 currently does not supply Dew Point so we have to calculate it in DoOutdoorTemp() + if (outdoortemp > -999) + DoOutdoorTemp(ConvertUnits.TempCToUser(outdoortemp), dataLastRead); + + // Same for extra T/H sensors + for (var i = 1; i <= 8; i++) + { + if (ExtraHum[i] > 0) + { + var dp = MeteoLib.DewPoint(ConvertUnits.UserTempToC(ExtraTemp[i]), ExtraHum[i]); + ExtraDewPoint[i] = ConvertUnits.TempCToUser(dp); + } + } + + if (gustLast > -999 && windSpeedLast > -999 && windDirLast > -999) + { + DoWind(gustLast, windDirLast, windSpeedLast, dataLastRead); + } + + if (rainLast > -999 && rainRateLast > -999) + { + DoRain(rainLast, rainRateLast, dataLastRead); + } - // at the start of every 20 minutes to trigger battery status check - if ((minute % 20) == 0 && !cumulus.cancellationToken.IsCancellationRequested) + if (outdoortemp > -999) + { + DoWindChill(windchill, dataLastRead); + DoApparentTemp(dataLastRead); + DoFeelsLike(dataLastRead); + DoHumidex(dataLastRead); + DoCloudBaseHeatIndex(dataLastRead); + + if (cumulus.StationOptions.CalculateSLP) { - GetSensorIdsNew(); + var abs = cumulus.Calib.Press.Calibrate(StationPressure); + var slp = MeteoLib.GetSeaLevelPressure(AltitudeM(cumulus.Altitude), ConvertUnits.UserPressToMB(abs), ConvertUnits.UserTempToC(OutdoorTemperature), cumulus.Latitude); + DoPressure(ConvertUnits.PressMBToUser(slp), dataLastRead); } + } + + DoForecast("", false); + + cumulus.BatteryLowAlarm.Triggered = batteryLow; + + UpdateStatusPanel(dataLastRead); + UpdateMQTT(); + + dataReceived = true; + DataStopped = false; + cumulus.DataStoppedAlarm.Triggered = false; + + var minute = DateTime.Now.Minute; + if (minute != lastMinute) + { + lastMinute = minute; // every day dump the clock drift at midday each day if (minute == 0 && DateTime.Now.Hour == 12) @@ -333,203 +456,6 @@ public override string GetEcowittVideoUrl() } - private Discovery DiscoverGW1000() - { - // We only want unique IP addresses - var discovered = new Discovery(); - const int broadcastPort = 46000; - - try - { - using var client = new UdpClient(); - var recvEp = new IPEndPoint(0, 0); - var sendEp = new IPEndPoint(IPAddress.Broadcast, broadcastPort); - var sendBytes = new byte[] { 0xff, 0xff, 0x12, 0x00, 0x04, 0x16 }; - - - // Get the primary IP address - var myIP = Utils.GetIpWithDefaultGateway(); - cumulus.LogDebugMessage($"Using local IP address {myIP} to discover the Ecowitt device"); - - // bind the cient to the primary address - broadcast does not work with .Any address :( - client.Client.Bind(new IPEndPoint(myIP, broadcastPort)); - // time out listening after 1.5 second - client.Client.ReceiveTimeout = 1500; - - // we are going to attempt discovery twice - var retryCount = 1; - do - { - cumulus.LogDebugMessage("Discovery Run #" + retryCount); - // each time we wait 1.5 second for any responses - var endTime = DateTime.Now.AddSeconds(1.5); - - try - { - // Send the request - client.Send(sendBytes, sendBytes.Length, sendEp); - - do - { - try - { - // get a response - var recevBuffer = client.Receive(ref recvEp); - - // sanity check the response size - we may see our request back as a receive packet - if (recevBuffer.Length > 20) - { - string ipAddr = $"{recevBuffer[11]}.{recevBuffer[12]}.{recevBuffer[13]}.{recevBuffer[14]}"; - var macArr = new byte[6]; - - Array.Copy(recevBuffer, 5, macArr, 0, 6); - var macHex = BitConverter.ToString(macArr).Replace('-', ':'); - - var nameLen = recevBuffer[17]; - var nameArr = new byte[nameLen]; - Array.Copy(recevBuffer, 18, nameArr, 0, nameLen); - var name = Encoding.UTF8.GetString(nameArr, 0, nameArr.Length); - - if (ipAddr.Split(dotSeparator, StringSplitOptions.RemoveEmptyEntries).Length == 4) - { - IPAddress ipAddr2; - if (IPAddress.TryParse(ipAddr, out ipAddr2)) - { - cumulus.LogDebugMessage($"Discovered Ecowitt device: {name}, IP={ipAddr}, MAC={macHex}"); - if (!discovered.IP.Contains(ipAddr) && !discovered.Mac.Contains(macHex)) - { - discovered.Name.Add(name); - discovered.IP.Add(ipAddr); - discovered.Mac.Add(macHex); - } - } - } - else - { - cumulus.LogDebugMessage($"Discovered an unsupported device: {name}, IP={ipAddr}, MAC={macHex}"); - } - } - } - catch - { - //do nothing - } - - } while (DateTime.Now < endTime); - } - catch (Exception ex) - { - cumulus.LogMessage("DiscoverGW1000: Error sending discovery request"); - cumulus.LogErrorMessage("DiscoverGW1000: Error: " + ex.Message); - } - retryCount++; - - } while (retryCount <= 2); - } - catch (Exception ex) - { - cumulus.LogErrorMessage("An error occurred during Ecowitt auto-discovery"); - cumulus.LogMessage("Error: " + ex.Message); - } - - return discovered; - } - - private void DoDiscovery() - { - if (cumulus.Gw1000AutoUpdateIpAddress || string.IsNullOrWhiteSpace(cumulus.Gw1000IpAddress)) - { - string msg; - cumulus.LogMessage("Running Ecowitt Local API auto-discovery..."); - cumulus.LogMessage($"Current IP address={cumulus.Gw1000IpAddress}, current MAC={cumulus.Gw1000MacAddress}"); - - var discoveredDevices = DiscoverGW1000(); - - if (discoveredDevices.IP.Count == 0) - { - // We didn't find anything on the network - msg = "Failed to discover any Ecowitt devices"; - cumulus.LogWarningMessage(msg); - Cumulus.LogConsoleMessage(msg, ConsoleColor.DarkYellow, true); - return; - } - else if (discoveredDevices.IP.Count == 1 && (string.IsNullOrEmpty(macaddr) || discoveredDevices.Mac[0] == macaddr)) - { - cumulus.LogDebugMessage("Discovered one Ecowitt device"); - // If only one device is discovered, and its MAC address matches (or our MAC is blank), then just use it - if (cumulus.Gw1000IpAddress != discoveredDevices.IP[0]) - { - cumulus.LogWarningMessage("Discovered a new IP address for the Ecowitt device that does not match our current one"); - cumulus.LogMessage($"Changing previous IP address: {ipaddr} to {discoveredDevices.IP[0]}"); - ipaddr = discoveredDevices.IP[0].Trim(); - cumulus.Gw1000IpAddress = ipaddr; - deviceModel = discoveredDevices.Name[0].Split('-')[0]; - deviceFirmware = discoveredDevices.Name[0].Split('-')[1].Split(' ')[1]; - if (discoveredDevices.Mac[0] != macaddr) - { - cumulus.Gw1000MacAddress = discoveredDevices.Mac[0].ToUpper(); - } - cumulus.WriteIniFile(); - } - else - { - cumulus.LogMessage("The discovered IP address for the GW1000 matches our current one"); - } - } - else if (discoveredDevices.Mac.Contains(macaddr)) - { - // Multiple devices discovered, but we have a MAC address match - - cumulus.LogDebugMessage("Matching Ecowitt MAC address found on the network"); - - var idx = discoveredDevices.Mac.IndexOf(macaddr); - deviceModel = discoveredDevices.Name[idx].Split('-')[0]; - deviceFirmware = discoveredDevices.Name[idx].Split('-')[1].Split(' ')[1]; - - if (discoveredDevices.IP[idx] != ipaddr) - { - cumulus.LogMessage("Discovered a new IP address for the Ecowitt device that does not match our current one"); - cumulus.LogMessage($"Changing previous IP address: {ipaddr} to {discoveredDevices.IP[idx]}"); - ipaddr = discoveredDevices.IP[idx]; - cumulus.Gw1000IpAddress = ipaddr; - cumulus.WriteIniFile(); - } - } - else - { - // Multiple devices discovered, and we do not have a clue! - - StringBuilder iplist = new(" discovered IPs = "); - msg = "Discovered more than one potential Ecowitt device."; - cumulus.LogWarningMessage(msg); - Cumulus.LogConsoleMessage(msg); - msg = "Please select the IP address from the list and enter it manually into the configuration"; - cumulus.LogMessage(msg); - Cumulus.LogConsoleMessage(msg); - - for (var i = 0; i < discoveredDevices.IP.Count; i++) - { - msg = $"Device={discoveredDevices.Name[i]}, IP={discoveredDevices.IP[i]}"; - Cumulus.LogConsoleMessage(msg); - cumulus.LogMessage(msg); - iplist.Append(discoveredDevices.IP[i] + " "); - } - cumulus.LogMessage(iplist.ToString()); - return; - } - } - - if (string.IsNullOrWhiteSpace(ipaddr)) - { - var msg = "No IP address configured or discovered for your GW1000, please remedy and restart Cumulus MX"; - cumulus.LogErrorMessage(msg); - Cumulus.LogConsoleMessage(msg); - return; - } - - return; - } - private string GetFirmwareVersion() { var response = "???"; @@ -582,771 +508,542 @@ private bool PrintSensorInfoNew(byte[] data, int idx) return batteryLow; } - private void GetLiveData() + private void ProcessCommonList(EcowittLocalApi.commonSensor[] sensors, DateTime dateTime) { - cumulus.LogDebugMessage("Reading live data"); + cumulus.LogDebugMessage($"ProcessCommonList: Processing {sensors.Length} sensors"); - byte[] data = null; // localApi.DoCommand(GW1000Api.Commands.CMD_GW1000_LIVEDATA); - - try + foreach (var sensor in sensors) { - if (null != data && data.Length > 16) + switch (sensor.id) { -#pragma warning disable S125 // Sections of code should not be commented out - /* - * debugging code with example data - * - //var hex = "FFFF27004601009D06220821A509270D02001707490A00B40B002F0C0069150001F07C16006317012A00324D00341900AA0E0000100000110000120000009D130000072C0D0000F8"; - var hex = "ffff27009c0100c806360827500927500200b107630a00710b00000c00001500000a2816000017002c2d2e28303332561c00c224361d00c325361e00c226361f00c3273621001b580059006200000012616609bbfc60011900240e00001000d31100d3120000022f130000022f0d00af6300684d6400cd4465006d4a66ff5b4e6700c74d6b00dc31002700190018001002550265000a0008002200160630"; - int NumberChars = hex.Length; - byte[] bytes = new byte[NumberChars / 2]; - for (int i = 0; i < NumberChars; i += 2) - bytes[i / 2] = Convert.ToByte(hex.Substring(i, 2), 16); - data = bytes; - */ -#pragma warning restore S125 // Sections of code should not be commented out - - // now decode it - Int16 tempInt16; - UInt16 tempUint16; - UInt32 tempUint32; - var idx = 5; - var dateTime = DateTime.Now; - var size = GW1000Api.ConvertBigEndianUInt16(data, 3); - - double windSpeedLast = -999, rainRateLast = -999, rainLast = -999, gustLast = -999; - int windDirLast = -999; - double outdoortemp = -999; - //double dewpoint - double windchill = -999; - - bool batteryLow = false; - - // We check the new value against what we have already, if older then ignore it! - double newLightningDistance = 999; - var newLightningTime = new DateTime(1900, 1, 1, 0, 0, 0, 0, DateTimeKind.Utc); + case "0x02": //Outdoor Temperature + if (sensor.valDbl.HasValue && cumulus.Gw1000PrimaryTHSensor == 0) + { + // do not process temperature here as if "MX calculates DP" is enabled, we have not yet read the humidity value. Have to do it at the end. + outdoortemp = sensor.valDbl.Value; + outdoortemp = sensor.unit == "C" ? ConvertUnits.TempCToUser(outdoortemp) : ConvertUnits.TempFToUser(outdoortemp); + } + break; + case "0x03": //Dew point + if (sensor.valDbl.HasValue && cumulus.Gw1000PrimaryTHSensor == 0 && !cumulus.StationOptions.CalculatedDP) + { + var temp = sensor.valDbl.Value; + temp = sensor.unit == "C" ? ConvertUnits.TempCToUser(temp) : ConvertUnits.TempFToUser(temp); - do - { - int chan; - switch (data[idx++]) + DoOutdoorDewpoint(temp, dateTime); + } + break; + case "3": //Feels like + // do nothing with this for now - MX calcuates feels like + break; + case "0x04": //Wind chill + if (sensor.valDbl.HasValue && cumulus.Gw1000PrimaryTHSensor == 0) { - case 0x01: //Indoor Temperature (℃) - tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx); - // user has mapped indoor temp to outdoor temp - if (cumulus.Gw1000PrimaryTHSensor == 99) - { - // do not process temperature here as if "MX calculates DP" is enabled, we have not yet read the humidity value. Have to do it at the end. - outdoortemp = tempInt16 / 10.0; - } - DoIndoorTemp(ConvertUnits.TempCToUser(tempInt16 / 10.0)); - idx += 2; - break; - case 0x02: //Outdoor Temperature (℃) - if (cumulus.Gw1000PrimaryTHSensor == 0) - { - tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx); - // do not process temperature here as if "MX calculates DP" is enabled, we have not yet read the humidity value. Have to do it at the end. - outdoortemp = tempInt16 / 10.0; - } - idx += 2; - break; - case 0x03: //Dew point (℃) - //tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx) - //dewpoint = tempInt16 / 10.0 - idx += 2; - break; - case 0x04: //Wind chill (℃) - if (cumulus.Gw1000PrimaryTHSensor == 0) - { - tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx); - windchill = tempInt16 / 10.0; - } - idx += 2; - break; - case 0x05: //Heat index (℃) - // cumulus calculates this - idx += 2; - break; - case 0x06: //Indoor Humidity(%) - // user has mapped indoor hum to outdoor hum - if (cumulus.Gw1000PrimaryTHSensor == 99) - { - DoOutdoorHumidity(data[idx], dateTime); - } - DoIndoorHumidity(data[idx]); - idx += 1; - break; - case 0x07: //Outdoor Humidity (%) - if (cumulus.Gw1000PrimaryTHSensor == 0) - { - DoOutdoorHumidity(data[idx], dateTime); - } - idx += 1; - break; - case 0x08: //Absolute Barometric (hPa) - tempUint16 = GW1000Api.ConvertBigEndianUInt16(data, idx); - StationPressure = ConvertUnits.PressMBToUser(tempUint16 / 10.0); - AltimeterPressure = ConvertUnits.PressMBToUser(MeteoLib.StationToAltimeter(tempUint16 / 10.0, AltitudeM(cumulus.Altitude))); - // Leave calculate SLP until the end as it depends on temperature - idx += 2; - break; - case 0x09: //Relative Barometric (hPa) - if (!cumulus.StationOptions.CalculateSLP) - { - tempUint16 = GW1000Api.ConvertBigEndianUInt16(data, idx); - DoPressure(ConvertUnits.PressMBToUser(tempUint16 / 10.0), dateTime); - } - idx += 2; - break; - case 0x0A: //Wind Direction (360°) - windDirLast = GW1000Api.ConvertBigEndianUInt16(data, idx); - idx += 2; - break; - case 0x0B: //Wind Speed (m/s) - windSpeedLast = ConvertUnits.WindMSToUser(GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0); - idx += 2; - break; - case 0x0C: // Gust speed (m/s) - gustLast = ConvertUnits.WindMSToUser(GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0); - idx += 2; - break; - case 0x0D: //Rain Event (mm) - if (cumulus.Gw1000PrimaryRainSensor == 0) - { - StormRain = ConvertUnits.RainMMToUser(GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0); - } - idx += 2; - break; - case 0x0E: //Rain Rate (mm/h) - if (cumulus.Gw1000PrimaryRainSensor == 0) - { - rainRateLast = ConvertUnits.RainMMToUser(GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0); - } - idx += 2; - break; - case 0x0F: //Rain Gain (mm) - idx += 2; - break; - case 0x10: //Rain Day (mm) - idx += 2; - break; - case 0x11: //Rain Week (mm) - idx += 2; - break; - case 0x12: //Rain Month (mm) - idx += 4; - break; - case 0x13: //Rain Year (mm) - if (cumulus.Gw1000PrimaryRainSensor == 0) - { - rainLast = ConvertUnits.RainMMToUser(GW1000Api.ConvertBigEndianUInt32(data, idx) / 10.0); - } - idx += 4; - break; - case 0x14: //Rain Totals (mm) - idx += 4; - break; - case 0x15: //Light (lux) - // Save the Lux value - LightValue = GW1000Api.ConvertBigEndianUInt32(data, idx) / 10.0; - // convert Lux to W/m² - approximately! - DoSolarRad((int) (LightValue * cumulus.SolarOptions.LuxToWM2), dateTime); - idx += 4; - break; - case 0x16: //UV (µW/cm²) - what use is this! - idx += 2; - break; - case 0x17: //UVI (0-15 index) - DoUV(data[idx], dateTime); - idx += 1; - break; - case 0x18: //Date and time - // does not appear to be implemented - idx += 6; - break; - case 0x19: //Day max wind(m/s) - idx += 2; - break; - case 0x1A: //Temperature 1(℃) - case 0x1B: //Temperature 2(℃) - case 0x1C: //Temperature 3(℃) - case 0x1D: //Temperature 4(℃) - case 0x1E: //Temperature 5(℃) - case 0x1F: //Temperature 6(℃) - case 0x20: //Temperature 7(℃) - case 0x21: //Temperature 8(℃) - chan = data[idx - 1] - 0x1A + 1; - tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx); - if (cumulus.Gw1000PrimaryTHSensor == chan) - { - outdoortemp = tempInt16 / 10.0; - } - DoExtraTemp(ConvertUnits.TempCToUser(tempInt16 / 10.0), chan); - idx += 2; - break; - case 0x22: //Humidity 1, 0-100% - case 0x23: //Humidity 2, 0-100% - case 0x24: //Humidity 3, 0-100% - case 0x25: //Humidity 4, 0-100% - case 0x26: //Humidity 5, 0-100% - case 0x27: //Humidity 6, 0-100% - case 0x28: //Humidity 7, 0-100% - case 0x29: //Humidity 8, 0-100% - chan = data[idx - 1] - 0x22 + 1; - if (cumulus.Gw1000PrimaryTHSensor == chan) - { - DoOutdoorHumidity(data[idx], dateTime); - } - DoExtraHum(data[idx], chan); - idx += 1; - break; - case 0x2B: //Soil Temperature1 (℃) - case 0x2D: //Soil Temperature2 (℃) - case 0x2F: //Soil Temperature3 (℃) - case 0x31: //Soil Temperature4 (℃) - case 0x33: //Soil Temperature5 (℃) - case 0x35: //Soil Temperature6 (℃) - case 0x37: //Soil Temperature7 (℃) - case 0x39: //Soil Temperature8 (℃) - case 0x3B: //Soil Temperature9 (℃) - case 0x3D: //Soil Temperature10 (℃) - case 0x3F: //Soil Temperature11 (℃) - case 0x41: //Soil Temperature12 (℃) - case 0x43: //Soil Temperature13 (℃) - case 0x45: //Soil Temperature14 (℃) - case 0x47: //Soil Temperature15 (℃) - case 0x49: //Soil Temperature16 (℃) - // figure out the channel number - chan = data[idx - 1] - 0x2B + 2; // -> 2,4,6,8... - chan /= 2; // -> 1,2,3,4... - tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx); - DoSoilTemp(ConvertUnits.TempCToUser(tempInt16 / 10.0), chan); - idx += 2; - break; - case 0x2C: //Soil Moisture1 (%) - case 0x2E: //Soil Moisture2 (%) - case 0x30: //Soil Moisture3 (%) - case 0x32: //Soil Moisture4 (%) - case 0x34: //Soil Moisture5 (%) - case 0x36: //Soil Moisture6 (%) - case 0x38: //Soil Moisture7 (%) - case 0x3A: //Soil Moisture8 (%) - case 0x3C: //Soil Moisture9 (%) - case 0x3E: //Soil Moisture10 (%) - case 0x40: //Soil Moisture11 (%) - case 0x42: //Soil Moisture12 (%) - case 0x44: //Soil Moisture13 (%) - case 0x46: //Soil Moisture14 (%) - case 0x48: //Soil Moisture15 (%) - case 0x4A: //Soil Moisture16 (%) - // figure out the channel number - chan = data[idx - 1] - 0x2C + 2; // -> 2,4,6,8... - chan /= 2; // -> 1,2,3,4... - DoSoilMoisture(data[idx], chan); - idx += 1; - break; - case 0x4C: //All sensor lowbatt 16 char - // This has been deprecated since v1.6.5 - now use CMD_READ_SENSOR_ID_NEW - idx += 16; - break; - case 0x2A: //PM2.5 Air Quality Sensor(μg/m³) - tempUint16 = GW1000Api.ConvertBigEndianUInt16(data, idx); - DoAirQuality(tempUint16 / 10.0, 1); - idx += 2; - break; - case 0x4D: //for pm25_ch1 - case 0x4E: //for pm25_ch2 - case 0x4F: //for pm25_ch3 - case 0x50: //for pm25_ch4 - chan = data[idx - 1] - 0x4D + 1; - tempUint16 = GW1000Api.ConvertBigEndianUInt16(data, idx); - DoAirQualityAvg(tempUint16 / 10.0, chan); - idx += 2; - break; - case 0x51: //PM2.5 ch_2 Air Quality Sensor(μg/m³) - case 0x52: //PM2.5 ch_3 Air Quality Sensor(μg/m³) - case 0x53: //PM2.5 ch_4 Air Quality Sensor(μg/m³) - chan = data[idx - 1] - 0x51 + 2; - tempUint16 = GW1000Api.ConvertBigEndianUInt16(data, idx); - DoAirQuality(tempUint16 / 10.0, chan); - idx += 2; - break; - case 0x58: //Leak ch1 - case 0x59: //Leak ch2 - case 0x5A: //Leak ch3 - case 0x5B: //Leak ch4 - chan = data[idx - 1] - 0x58 + 1; - DoLeakSensor(data[idx], chan); - idx += 1; - break; - case 0x60: //Lightning dist (1-40km) - // Sends a default value of 255km until the first strike is detected - newLightningDistance = data[idx] == 0xFF ? 999 : ConvertUnits.KmtoUserUnits(data[idx]); - idx += 1; - break; - case 0x61: //Lightning time (UTC) - // Sends a default value until the first strike is detected of 0xFFFFFFFF - tempUint32 = GW1000Api.ConvertBigEndianUInt32(data, idx); - if (tempUint32 == 0xFFFFFFFF) - { - newLightningTime = new DateTime(1900, 1, 1, 0, 0, 0, 0, DateTimeKind.Utc); - } - else - { - var dtDateTime = DateTime.UnixEpoch; - dtDateTime = dtDateTime.AddSeconds(tempUint32).ToLocalTime(); - newLightningTime = dtDateTime; - } - idx += 4; - break; - case 0x62: //Lightning strikes today - tempUint32 = GW1000Api.ConvertBigEndianUInt32(data, idx); - if (tempUint32 == 0 && dateTime.Minute == 59 && dateTime.Hour == 23) - { - // Ecowitt clock drift - if the count resets in the minute before midnight, ignore it until after midnight - } - else - { - LightningStrikesToday = (int) tempUint32; - } - idx += 4; - break; - // user temp = WH34 8 channel Soil or Water temperature sensors - case 0x63: // user temp ch1 (°C) - case 0x64: // user temp ch2 (°C) - case 0x65: // user temp ch3 (°C) - case 0x66: // user temp ch4 (°C) - case 0x67: // user temp ch5 (°C) - case 0x68: // user temp ch6 (°C) - case 0x69: // user temp ch7 (°C) - case 0x6A: // user temp ch8 (°C) - chan = data[idx - 1] - 0x63 + 1; - tempInt16 = GW1000Api.ConvertBigEndianInt16(data, idx); - if (cumulus.EcowittMapWN34[chan] == 0) // false = user temp, true = soil temp - { - DoUserTemp(ConvertUnits.TempCToUser(tempInt16 / 10.0), chan); - } - else - { - DoSoilTemp(ConvertUnits.TempCToUser(tempInt16 / 10.0), cumulus.EcowittMapWN34[chan]); - } - // Firmware version 1.5.9 uses 2 data bytes, 1.6.0+ uses 3 data bytes - if ((deviceModel.StartsWith("GW1000") && fwVersion >= new Version("1.6.0")) || !deviceModel.StartsWith("GW1000")) - { - var volts = TestBattery10V(data[idx + 2]); - if (volts <= 1.2) - { - batteryLow = true; - cumulus.LogWarningMessage($"WN34 channel #{chan} battery LOW = {volts}V"); - } - else - { - cumulus.LogDebugMessage($"WN34 channel #{chan} battery OK = {volts}V"); - } - idx += 3; - } - else - { - idx += 2; - } - break; - case 0x6B: //WH34 User temperature battery (8 channels) - No longer used in firmware 1.6.0+ - //Later version from ??? send an extended WH45 CO₂ data block - if (deviceModel.StartsWith("GW1000") && fwVersion < new Version("1.6.0")) - { - batteryLow = batteryLow || DoWH34BatteryStatus(data, idx); - idx += 8; - } - else - { - batteryLow = batteryLow || DoCO2DecodeNew(data, idx); - idx += 23; - } - break; - case 0x6C: // Heap size - has constant offset of +3692 to GW1100 HTTP value???? - StationFreeMemory = (int) GW1000Api.ConvertBigEndianUInt32(data, idx); - idx += 4; - break; - case 0x70: // WH45 CO₂ - batteryLow = batteryLow || DoCO2DecodeNew(data, idx); - idx += 16; - break; - case 0x71: // Ambient ONLY - AQI - // Not doing anything with this yet - //idx += 2; // SEEMS TO BE VARIABLE?! - cumulus.LogDebugMessage("Found a device 0x71 - Ambient AQI. No decode for this yet"); - // We will have lost our place now, so bail out - idx = size; - break; - case 0x72: // WH35 Leaf Wetness ch1 - case 0x73: // WH35 Leaf Wetness ch2 - case 0x74: // WH35 Leaf Wetness ch3 - case 0x75: // WH35 Leaf Wetness ch4 - case 0x76: // WH35 Leaf Wetness ch5 - case 0x77: // WH35 Leaf Wetness ch6 - case 0x78: // WH35 Leaf Wetness ch7 - case 0x79: // WH35 Leaf Wetness ch8 - chan = data[idx - 1] - 0x72 + 2; // -> 2,4,6,8... - chan /= 2; // -> 1,2,3,4... - DoLeafWetness(data[idx], chan); - idx += 1; - break; - case 0x7A: // Rain Priority - idx += 1; - break; - case 0x7B: // Radiation compensation - idx += 1; - break; - case 0x80: // Piezo Rain Rate - if (cumulus.Gw1000PrimaryRainSensor == 1) - { - rainRateLast = ConvertUnits.RainMMToUser(GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0); - } - idx += 2; - break; - case 0x81: // Piezo Rain Event - if (cumulus.Gw1000PrimaryRainSensor == 1) - { - StormRain = ConvertUnits.RainMMToUser(GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0); - } - idx += 2; - break; - case 0x82: // Piezo Hourly Rain (not used) - idx += 2; - break; - case 0x83: // Piezo Daily Rain - idx += 2; - break; - case 0x84: // Piezo Weekly Rain - idx += 2; - break; - case 0x85: // Piezo Monthly Rain - idx += 4; - break; - case 0x86: // Piezo Yearly Rain - if (cumulus.Gw1000PrimaryRainSensor == 1) - { - rainLast = ConvertUnits.RainMMToUser(GW1000Api.ConvertBigEndianUInt32(data, idx) / 10.0); - } - idx += 4; - break; - case 0x87: // Piezo Gain - doc says size = 2*10 ? - idx += 20; - break; - case 0x88: // Piezo Rain Reset Time - idx += 3; - break; - default: - cumulus.LogDebugMessage($"Error: Unknown sensor id found = {data[idx - 1]}, at position = {idx - 1}"); - // We will have lost our place now, so bail out - idx = size; - break; + windchill = sensor.valDbl.Value; + windchill = sensor.unit == "C" ? ConvertUnits.TempCToUser(windchill) : ConvertUnits.TempFToUser(windchill); + } + break; + case "0x05": //Heat index + // cumulus calculates this + break; + case "0x07": //Outdoor Humidity (%) + if (sensor.valInt.HasValue && cumulus.Gw1000PrimaryTHSensor == 0) + { + DoOutdoorHumidity(sensor.valInt.Value, dateTime); + } + break; + case "0x0A": //Wind Direction (360°) + if (sensor.valInt.HasValue) + { + windDirLast = sensor.valInt.Value; + } + break; + case "0x0B": //Wind Speed (val unit) + var arr = sensor.val.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out var valDbl)) + { + var spd = arr[1] switch + { + "km/h" => ConvertUnits.WindKPHToUser(valDbl), + "m/s" => ConvertUnits.WindMSToUser(valDbl), + "mph" => ConvertUnits.WindMPHToUser(valDbl), + "knots" => ConvertUnits.WindKnotsToUser(valDbl), + _ => -999 + }; + + if (spd >= 0) + { + windSpeedLast = spd; + } + } + break; + case "0x0C": // Gust speed (val unit) + arr = sensor.val.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out valDbl)) + { + var spd = arr[1] switch + { + "km/h" => ConvertUnits.WindKPHToUser(valDbl), + "m/s" => ConvertUnits.WindMSToUser(valDbl), + "mph" => ConvertUnits.WindMPHToUser(valDbl), + "knots" => ConvertUnits.WindKnotsToUser(valDbl), + _ => -999 + }; + + if (spd >= 0) + { + gustLast = spd; + } + } + break; + case "0x15": //Light (value unit) + arr = sensor.val.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out valDbl)) + { + var light = arr[1] switch + { + "fc" => valDbl * 0.015759751708199, + "lux" => valDbl * cumulus.SolarOptions.LuxToWM2, // convert Lux to W/m² - approximately! + "W/m2" => valDbl, + _ => -999 + }; + + LightValue = valDbl; + if (light >= 0) + { + DoSolarRad((int) light, dateTime); + } } - } while (idx < size); + break; + case "0x17": //UVI (0-15 index) + if (sensor.valDbl.HasValue) + { + DoUV(sensor.valDbl.Value, dateTime); + } + break; - // Some debugging info - cumulus.LogDebugMessage($"LiveData: Wind Decode >> Last={windSpeedLast:F1}, LastDir={windDirLast}, Gust={gustLast:F1}, (MXAvg={WindAverage:F1})"); + case "0x19": // max wind today (value unit) + // not used + break; - // Now do the stuff that requires more than one input parameter + default: + cumulus.LogDebugMessage($"Error: Unknown common_list sensor id found = {sensor.id}"); + break; + } + } + + // Some debugging info + cumulus.LogDebugMessage($"LiveData: Wind Decode >> Last={windSpeedLast:F1}, LastDir={windDirLast}, Gust={gustLast:F1}, (MXAvg={WindAverage:F1})"); + + } - // Only set the lightning time/distance if it is newer than what we already have - the GW1000 seems to reset this value - if (newLightningTime > LightningTime) + private void ProcessWh25(EcowittLocalApi.wh25Sensor[] sensors, DateTime dateTime) + { + foreach (var sensor in sensors) + { + // Indoor Temperature + try + { + var temp = sensor.intemp; + temp = sensor.unit == "C" ? ConvertUnits.TempCToUser(temp) : ConvertUnits.TempFToUser(temp); + + // user has mapped indoor temp to outdoor temp + if (cumulus.Gw1000PrimaryTHSensor == 99) { - LightningTime = newLightningTime; - if (newLightningDistance < 999) - LightningDistance = newLightningDistance; + // do not process temperature here as if "MX calculates DP" is enabled, we have not yet read the humidity value. Have to do it at the end. + DoOutdoorTemp(temp, dateTime); } + DoIndoorTemp(temp); + } + catch (Exception ex) + { + //TODO: log a message + } - // Process outdoor temperature here, as GW1000 currently does not supply Dew Point so we have to calculate it in DoOutdoorTemp() - if (outdoortemp > -999) - DoOutdoorTemp(ConvertUnits.TempCToUser(outdoortemp), dateTime); + // Indoor Humidity + try + { + var hum = sensor.inhumiInt ?? 0; + // user has mapped indoor hum to outdoor hum + if (cumulus.Gw1000PrimaryTHSensor == 99) + { + DoOutdoorHumidity(hum, dateTime); + } + DoIndoorHumidity(hum); + } + catch (Exception ex) + { + //TODO: log a message + } - // Same for extra T/H sensors - for (var i = 1; i <= 8; i++) + // Pressure + try + { + if (sensor.rel != null && !cumulus.StationOptions.CalculateSLP) { - if (ExtraHum[i] > 0) + var arr = sensor.rel.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out var val)) { - var dp = MeteoLib.DewPoint(ConvertUnits.UserTempToC(ExtraTemp[i]), ExtraHum[i]); - ExtraDewPoint[i] = ConvertUnits.TempCToUser(dp); + var slp = arr[1] switch + { + "hPa" => ConvertUnits.PressKPAToUser(val / 10), + "inHg" => ConvertUnits.PressINHGToUser(val), + "mmHg" => ConvertUnits.PressINHGToUser(val * 25.4), + _ => -999, + }; + + if (slp > 0) + { + DoPressure(slp, dateTime); + } } } - - if (gustLast > -999 && windSpeedLast > -999 && windDirLast > -999) + if (sensor.abs != null) { - DoWind(gustLast, windDirLast, windSpeedLast, dateTime); + var arr = sensor.abs.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out var val)) + { + var abs = arr[1] switch + { + "hPa" => ConvertUnits.PressKPAToUser(val / 10), + "inHg" => ConvertUnits.PressINHGToUser(val), + "mmHg" => ConvertUnits.PressINHGToUser(val * 25.4), + _ => -999 + }; + + if (abs > 0) + { + StationPressure = abs; + AltimeterPressure = ConvertUnits.PressMBToUser(MeteoLib.StationToAltimeter(abs, AltitudeM(cumulus.Altitude))); + // Leave calculate SLP until the end as it depends on temperature + } + } } + } + catch (Exception ex) + { + //TODO: log a message + } - if (rainLast > -999 && rainRateLast > -999) + // TODO: battery status + } + } + + private void ProcessRain(EcowittLocalApi.commonSensor[] sensors, DateTime dateTime) + { + foreach (var sensor in sensors) + { + //Rain Event (val unit) + try + { + var arr = sensor.val.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out var val)) { - DoRain(rainLast, rainRateLast, dateTime); + var evnt = arr[1] switch + { + "mm" => ConvertUnits.RainMMToUser(val), + "in" => ConvertUnits.RainINToUser(val), + _ => -999 + }; + + if (evnt >= 0) + { + StormRain = evnt; + } } + } + catch (Exception ex) + { + //TODO: log a message + } - if (outdoortemp > -999) + //Rain Rate (val unit/h) + try + { + var arr = sensor.val.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out var val)) { - DoWindChill(windchill, dateTime); - DoApparentTemp(dateTime); - DoFeelsLike(dateTime); - DoHumidex(dateTime); - DoCloudBaseHeatIndex(dateTime); + var rate = arr[1] switch + { + "mm/Hr" => ConvertUnits.RainMMToUser(val), + "in/Hr" => ConvertUnits.RainINToUser(val), + _ => -999 + }; - if (cumulus.StationOptions.CalculateSLP) + if (rate >= 0) { - var abs = cumulus.Calib.Press.Calibrate(StationPressure); - var slp = MeteoLib.GetSeaLevelPressure(AltitudeM(cumulus.Altitude), ConvertUnits.UserPressToMB(abs), ConvertUnits.UserTempToC(OutdoorTemperature), cumulus.Latitude); - DoPressure(ConvertUnits.PressMBToUser(slp), dateTime); + rainRateLast = rate; } } + } + catch (Exception ex) + { + //TODO: log a message + } - DoForecast("", false); - - cumulus.BatteryLowAlarm.Triggered = batteryLow; + //Rain Year (val unit) + try + { + // TODO: battery status - UpdateStatusPanel(dateTime); - UpdateMQTT(); + var arr = sensor.val.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out var val)) + { + var yr = arr[1] switch + { + "mm/Hr" => ConvertUnits.RainMMToUser(val), + "in/Hr" => ConvertUnits.RainINToUser(val), + _ => -999 + }; - dataReceived = true; - DataStopped = false; - cumulus.DataStoppedAlarm.Triggered = false; + if (yr >= 0) + { + rainLast = yr; + } + } } - else + catch (Exception ex) { - cumulus.LogWarningMessage("GetLiveData: Invalid response"); + //TODO: log a message } } - catch (Exception ex) + } + + private void ProcessLightning(EcowittLocalApi.lightningSensor[] sensors, DateTime dateTime) + { + foreach (var sensor in sensors) { - cumulus.LogErrorMessage("GetLiveData: Error - " + ex.Message); + try + { + //Lightning dist (1-40km) + if (sensor.distanceVal.HasValue && sensor.distanceUnit != null) + { + // Sends a default value of 255km until the first strike is detected + if (sensor.distanceVal.Value > 254.9) + { + newLightningDistance = 999; + } + else + { + newLightningDistance = ConvertUnits.KmtoUserUnits(sensor.distanceVal.Value); + } + } + + //Lightning time (UTC) + if (!string.IsNullOrEmpty(sensor.timestamp)) + { + // oh my god, it sends the time as "MM/dd/yyyy HH: mm: ss" + // TODO: what is default time if not strikes detecetd yet? + var arr = sensor.timestamp.Split(' '); + var date = arr[0].Split('/'); + + //if (sensor.timestamp == "default string") + //{ + // newLightningTime = new DateTime(1900, 1, 1, 0, 0, 0, 0, DateTimeKind.Utc); + //} + //else + //{ + newLightningTime = new DateTime( + int.Parse(date[2]), + int.Parse(date[0]), + int.Parse(date[1]), + int.Parse(arr[1][0..^1]), + int.Parse(arr[2][0..^1]), + int.Parse(arr[3][0..^1]), + 0, DateTimeKind.Utc); + //} + } + + //Lightning strikes today + if (sensor.count.HasValue) + { + if (sensor.count == 0 && dateTime.Minute == 59 && dateTime.Hour == 23) + { + // Ecowitt clock drift - if the count resets in the minute before midnight, ignore it until after midnight + } + else + { + LightningStrikesToday = sensor.count.Value; + } + } + } + catch (Exception ex) + { + //TODO: log a message + } } } - private void GetSystemInfo(bool driftOnly) + private void ProcessCo2(EcowittLocalApi.co2Sensor[] sensors, DateTime dateTime) { - cumulus.LogMessage("Reading Ecowitt system info"); + cumulus.LogDebugMessage("WH45 CO₂: Decoding..."); + + foreach (var sensor in sensors) + { + try + { + if (sensor.temp.HasValue && !string.IsNullOrEmpty(sensor.unit)) + { + CO2_temperature = sensor.unit == "C" ? ConvertUnits.TempCToUser(sensor.temp.Value) : ConvertUnits.TempFToUser(sensor.temp.Value); + } + if (sensor.humidityVal.HasValue) + { + // humidty sent as "value%" + CO2_humidity = sensor.humidityVal.Value; + } + if (sensor.PM25.HasValue) + { + CO2_pm2p5 = sensor.PM25.Value; + } + if (sensor.PM10.HasValue) + { + CO2_pm10 = sensor.PM10.Value; + } + // HTTP protocol does not send a pm 24 hour values :( On useless AQI values + if (sensor.CO2.HasValue) + { + CO2 = sensor.CO2.Value; + } + if (sensor.CO2_24H.HasValue) + { + CO2_24h = sensor.CO2_24H.Value; + } + + // TODO: Battery status + } + catch (Exception ex) + { + cumulus.LogExceptionMessage(ex, "ProcessCo2: Error"); + } + } + CO2_pm2p5_aqi = GetAqi(AqMeasure.pm2p5, CO2_pm2p5); + CO2_pm10_aqi = GetAqi(AqMeasure.pm10, CO2_pm10); } - private void GetPiezoRainData() + private void ProcessChPm25(EcowittLocalApi.ch_pm25Sensor[] sensors) { - cumulus.LogDebugMessage("GetPiezoRainData: Reading piezo rain data"); - - byte[] data = null; //localApi.DoCommand(GW1000Api.Commands.CMD_READ_RAIN); - - - // expected response - units mm - // 0 - 0xff - header - // 1 - 0xff - header - // 2 - 0x57 - rain data - // 3-4 - size(2) - // - // Field Value - // - data size - // - // 0E = rain rate - // - data(2) - // 10 = rain day - // - data(4) - // 11 = rain week - // - data(4) - // 12 = rain month - // - data(4) - // 13 = rain year - // - data(4) - // 0D = rain event - // - data(2) - // 0F = rain gain - // - data(2) - // 80 = piezo rain rate - // - data(2) - // 83 = piezo rain day - // - data(4) - // 84 = piezo rain week - // - data(4) - // 85 = piezo rain month - // - data(4) - // 86 = piezo rain year - // - data(4) - // 81 = piezo rain event - // - data(2) - // 87 = piezo gain 0-9 - // - data(2x10) - // 88 = rain reset time (hr, day [sun-0], month [jan=0]) - // - data(3) - // 7A = primary rain selection (0=No sensor, 1=Tipper, 2=Piezo) - // - data(1) - // 7B = solar gain compensation - // - data(1) - // 85 - checksum - - //data = new byte[] { 0xFF, 0xFF, 0x57, 0x00, 0x54, 0x0E, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x11, 0x00, 0x00, 0x00, 0x00, 0x12, 0x00, 0x00, 0x02, 0xF2, 0x13, 0x00, 0x00, 0x0B, 0x93, 0x0D, 0x00, 0x00, 0x0F, 0x00, 0x64, 0x80, 0x00, 0x00, 0x83, 0x00, 0x00, 0x00, 0x00, 0x84, 0x00, 0x00, 0x00, 0x00, 0x85, 0x00, 0x00, 0x01, 0xDE, 0x86, 0x00, 0x00, 0x0B, 0xF2, 0x81, 0x00, 0x00, 0x87, 0x00, 0x64, 0x00, 0x64, 0x00, 0x64, 0x00, 0x64, 0x00, 0x64, 0x00, 0x64, 0x00, 0x64, 0x00, 0x64, 0x00, 0x64, 0x00, 0x64, 0x88, 0x00, 0x00, 0x00, 0xF7 } - - if (data == null) + foreach (var sensor in sensors) { - cumulus.LogErrorMessage("GetPiezoRainData: Nothing returned from Read Rain!"); - return; + if (sensor.channel.HasValue && sensor.PM25.HasValue) + { + DoAirQuality(sensor.PM25.Value, sensor.channel.Value); + } + + // TODO: Battery status } + } - if (data.Length < 8) + private void ProcessLeak(EcowittLocalApi.ch_leakSensor[] sensors) + { + foreach (var sensor in sensors) { - cumulus.LogErrorMessage("GetPiezoRainData: Unexpected response to Read Rain!"); - return; + if (sensor.channel.HasValue && !string.IsNullOrEmpty(sensor.status)) + { + var val = sensor.status == "NORMAL" ? 0 : 1; + DoLeakSensor(val, sensor.channel.Value); + + // TODO: Battery status + } } - try + } + + + private void ProcessExtraTempHum(EcowittLocalApi.tempHumSensor[] sensors, DateTime dateTime) + { + foreach (var sensor in sensors) { - // There are reports of different sized messages from different gateways, - // so parse it sequentially like the live data rather than using fixed offsets - var idx = 5; - var size = GW1000Api.ConvertBigEndianUInt16(data, 3); - double? rRate = null; - double? rain = null; - - do + if (sensor.temp.HasValue) { - switch (data[idx++]) + DoExtraTemp(sensor.temp.Value, sensor.channel); + } + + if (sensor.humidityVal.HasValue) + { + if (cumulus.Gw1000PrimaryTHSensor == sensor.channel) { - // all the two byte values we are ignoring - case 0x0E: // rain rate - case 0x0D: // rain event - case 0x0F: // rain gain - case 0x81: // piezo rain event - idx += 2; - break; - // all the four byte values we are ignoring - case 0x10: // rain day - case 0x11: // rain week - case 0x12: // rain month - case 0x13: // rain year - case 0x83: // piezo rain day - case 0x84: // piezo rain week - case 0x85: // piezo rain month - idx += 4; - break; - case 0x80: // piezo rain rate - if (cumulus.StationOptions.UseRainForIsRaining == 2 && cumulus.Gw1000PrimaryRainSensor != 1) - { - IsRaining = GW1000Api.ConvertBigEndianUInt16(data, idx) > 0; - cumulus.IsRainingAlarm.Triggered = IsRaining; - } - else - { - rRate = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; - } - idx += 2; - break; - case 0x86: // piezo rain year - if (cumulus.Gw1000PrimaryRainSensor == 1) - rain = GW1000Api.ConvertBigEndianUInt32(data, idx) / 10.0; - idx += 4; - break; - case 0x87: // piezo gain 0-9 - idx += 20; - break; - case 0x88: // rain reset time -#if DEBUG - cumulus.LogDebugMessage($"GetPiezoRainData: Rain reset times - hour:{data[idx++]}, day:{data[idx++]}, month:{data[idx++]}"); -#else - idx += 3; -#endif - break; - case 0x7A: // Preferred rain sensor on station - var sensor = data[idx++]; -#if DEBUG - if (sensor == 0) - cumulus.LogDebugMessage("GetPiezoRainData: No rain sensor available"); - else if (sensor == 1) - cumulus.LogDebugMessage("GetPiezoRainData: Traditional rain sensor selected"); - else if (sensor == 2) - cumulus.LogDebugMessage("GetPiezoRainData: Piezo rain sensor selected"); - else - cumulus.LogDebugMessage("GetPiezoRainData: Unkown rain sensor selection value = " + sensor); -#endif - break; - case 0x7B: // Solar gain compensation -#if DEBUG - cumulus.LogDebugMessage($"GetPiezoRainData: Solar gain compensation = {(data[idx] == '0' ? "disabled" : "enabled")}"); -#endif - idx += 1; - break; - default: - cumulus.LogDebugMessage($"GetPiezoRainData: Error: Unknown value type found = {data[idx - 1]}, at position = {idx - 1}"); - // We will have lost our place now, so bail out - idx = size; - break; + DoOutdoorHumidity(sensor.humidityVal.Value, dateTime); } + DoExtraHum(sensor.humidityVal.Value, sensor.channel); + } + + // TODO: battery status + } + } - } while (idx < size); + private void ProcessUserTemp(EcowittLocalApi.tempHumSensor[] sensors, DateTime dateTime) + { + // user temp = WH34 8 channel Soil or Water temperature sensors - if (cumulus.Gw1000PrimaryRainSensor == 1) + foreach (var sensor in sensors) + { + if (sensor.temp.HasValue) { - if (rRate.HasValue && rain.HasValue) + var val = sensor.unit == "C" ? ConvertUnits.TempCToUser(sensor.temp.Value) : ConvertUnits.TempFToUser(sensor.temp.Value); + if (cumulus.EcowittMapWN34[sensor.channel] == 0) // false = user temp, true = soil temp { -#if DEBUG - cumulus.LogDebugMessage($"GetPiezoRainData: Rain Year: {rain:f1} mm, Rate: {rRate:f1} mm/hr"); -#endif - DoRain(ConvertUnits.RainMMToUser(rain.Value), ConvertUnits.RainMMToUser(rRate.Value), DateTime.Now); + DoUserTemp(val, sensor.channel); } else { - cumulus.LogErrorMessage("GetPiezoRainData: Error, no piezo rain data found in the response"); + DoSoilTemp(val, cumulus.EcowittMapWN34[sensor.channel]); } + + // TODO: Battery status } } - catch (Exception ex) + } + + private void ProcessSoilMoisture(EcowittLocalApi.tempHumSensor[] sensors) + { + foreach (var sensor in sensors) { - cumulus.LogErrorMessage("GetPiezoRainData: Error processing Rain Info: " + ex.Message); + if (sensor.humidityVal.HasValue) + { + DoSoilMoisture(sensor.humidityVal.Value, sensor.channel); + } + + // TODO: Battery status } } - private bool DoCO2DecodeNew(byte[] data, int index) + private void ProcessSoilTemp(EcowittLocalApi.tempHumSensor[] sensors) { - bool batteryLow = false; - int idx = index; - cumulus.LogDebugMessage("WH45 CO₂ New: Decoding..."); - - try + foreach (var sensor in sensors) { - CO2_temperature = ConvertUnits.TempCToUser(GW1000Api.ConvertBigEndianInt16(data, idx) / 10.0); - idx += 2; - CO2_humidity = data[idx++]; - CO2_pm10 = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; - idx += 2; - CO2_pm10_24h = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; - idx += 2; - CO2_pm2p5 = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; - idx += 2; - CO2_pm2p5_24h = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; - idx += 2; - CO2 = GW1000Api.ConvertBigEndianUInt16(data, idx); - idx += 2; - CO2_24h = GW1000Api.ConvertBigEndianUInt16(data, idx); - idx += 2; - CO2_pm1 = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; - idx += 2; - CO2_pm1_24h = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; - idx += 2; - CO2_pm4 = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; - idx += 2; - CO2_pm4_24h = GW1000Api.ConvertBigEndianUInt16(data, idx) / 10.0; - idx += 2; - var msg = $"WH45 CO₂ New: temp={CO2_temperature.ToString(cumulus.TempFormat)}, hum={CO2_humidity}, pm10={CO2_pm10:F1}, pm10_24h={CO2_pm10_24h:F1}, pm2.5={CO2_pm2p5:F1}, pm2.5_24h={CO2_pm2p5_24h:F1}, CO₂={CO2}, CO₂_24h={CO2_24h}, pm1={CO2_pm1:F1}, pm1_24h={CO2_pm1_24h:F1}, pm4={CO2_pm4:F1}, pm4_24h={CO2_pm4_24h:F1}"; - var batt = TestBattery3(data[idx]); - batteryLow = batt == "Low"; - msg += $", Battery={batt}"; - cumulus.LogDebugMessage(msg); - - CO2_pm2p5_aqi = GetAqi(AqMeasure.pm2p5, CO2_pm2p5); - CO2_pm2p5_24h_aqi = GetAqi(AqMeasure.pm2p5h24, CO2_pm2p5_24h); - CO2_pm10_aqi = GetAqi(AqMeasure.pm10, CO2_pm10); - CO2_pm10_24h_aqi = GetAqi(AqMeasure.pm10h24, CO2_pm10_24h); + if (sensor.temp.HasValue) + { + var val = sensor.unit == "C" ? ConvertUnits.TempCToUser(sensor.temp.Value) : ConvertUnits.TempFToUser(sensor.temp.Value); + DoSoilTemp(val, sensor.channel); + } + + // TODO: Battery status } - catch (Exception ex) + + } + + private void ProcessLeafWet(EcowittLocalApi.tempHumSensor[] sensors) + { + foreach (var sensor in sensors) { - cumulus.LogExceptionMessage(ex, "DoCO2DecodeNew: Error"); + if (sensor.humidityVal.HasValue) + { + DoLeafWetness(sensor.humidityVal.Value, sensor.channel); + } + + // TODO: Battery status } - return batteryLow; } + + private void GetSystemInfo(bool driftOnly) + { + cumulus.LogMessage("Reading Ecowitt system info"); + + } + + private bool DoWH34BatteryStatus(byte[] data, int index) { // No longer used in firmware 1.6.0+ @@ -1402,21 +1099,6 @@ private static string TestBatteryWh40(byte value, double volts) return volts > 1.2 ? "OK" : "Low"; } - - private sealed class Discovery - { - public List IP { get; set; } - public List Name { get; set; } - public List Mac { get; set; } - - public Discovery() - { - IP = []; - Name = []; - Mac = []; - } - } - private void DataTimeout(object source, ElapsedEventArgs e) { if (dataReceived) @@ -1435,7 +1117,6 @@ private void DataTimeout(object source, ElapsedEventArgs e) } cumulus.DataStoppedAlarm.LastMessage = $"No data received from the GW1000 for {tmrDataWatchdog.Interval / 1000} seconds"; cumulus.DataStoppedAlarm.Triggered = true; - DoDiscovery(); } } } diff --git a/CumulusMX/EcowittLocalApi.cs b/CumulusMX/EcowittLocalApi.cs index 52628590..0fa005f3 100644 --- a/CumulusMX/EcowittLocalApi.cs +++ b/CumulusMX/EcowittLocalApi.cs @@ -10,7 +10,6 @@ namespace CumulusMX internal sealed class EcowittLocalApi : IDisposable { private readonly Cumulus cumulus; - private string ipAddress = null; public EcowittLocalApi(Cumulus cumul) { @@ -148,14 +147,12 @@ public liveData GetLiveData(CancellationToken token) int responseCode; int retries = 3; - string ip; + string ip = cumulus.Gw1000IpAddress; int retry = 1; - ip = cumulus.DavisOptions.IPAddr; - do { - var url = $"http://{ip}/get_livedata_info?"; + var url = $"http://{ip}/get_livedata_info"; // we want to do this synchronously, so .Result using (var response = cumulus.MyHttpClient.GetAsync(url, token).Result) @@ -180,7 +177,7 @@ public liveData GetLiveData(CancellationToken token) Cumulus.LogConsoleMessage(" - No Live data available"); return null; } - else if (responseBody.StartsWith("{ \"common")) // sanity check + else if (responseBody.StartsWith('{')) // sanity check { // Convert JSON string to an object liveData json = responseBody.FromJson(); @@ -242,7 +239,7 @@ private enum commonSensorTypes public class commonSensor { - public int id { get; set; } + public string id { get; set; } public string val { get; set; } public string? unit { get; set; } public double? battery { get; set; } @@ -275,7 +272,7 @@ public double? valDbl public class tempHumSensor { public int channel { get; set; } - public int battery { get; set; } + public int? battery { get; set; } public double? temp { get; set; } public string? humidity { get; set; } public string? unit { get; set; } @@ -284,9 +281,7 @@ public int? humidityVal { get { - if (humidity.EndsWith('%')) - humidity = humidity[0..^1]; - return int.TryParse(humidity, out int result) ? result : null; + return int.TryParse(humidity[0..^1], out int result) ? result : null; } } } @@ -299,7 +294,7 @@ public class wh25Sensor public string inhumi { get; set; } public string abs { get; set; } public string rel { get; set; } - public int battery { get; set; } + public int? battery { get; set; } public int? inhumiInt { @@ -314,8 +309,8 @@ public class lightningSensor { public string distance { get; set; } public string timestamp { get; set; } - public int count { get; set; } - public int battery { get; set; } + public int? count { get; set; } + public int? battery { get; set; } public double? distanceVal { @@ -326,7 +321,7 @@ public double? distanceVal } } - public string? distanceUnit + public string distanceUnit { get { @@ -338,34 +333,42 @@ public string? distanceUnit public class co2Sensor { - public double temp { get; set; } + public double? temp { get; set; } public string unit { get; set; } public string humidity { get; set; } - public double PM25 { get; set; } - public double PM25_RealAQI { get; set; } - public double PM25_24HAQI { get; set; } - public double PM10 { get; set; } - public double PM10_RealAQI { get; set; } - public double PM10_24HAQI { get; set; } - public int CO2 { get; set; } - public int CO2_24H { get; set; } - public int battery { get; set; } + public double? PM25 { get; set; } + public double? PM25_RealAQI { get; set; } + public double? PM25_24HAQI { get; set; } + public double? PM10 { get; set; } + public double? PM10_RealAQI { get; set; } + public double? PM10_24HAQI { get; set; } + public int? CO2 { get; set; } + public int? CO2_24H { get; set; } + public int? battery { get; set; } + + public int? humidityVal + { + get + { + return int.TryParse(humidity[0..^1], out int result) ? result : null; + } + } } public class ch_pm25Sensor { - public int channel { get; set; } - public double PM25 { get; set; } - public double PM25_RealAQI { get; set; } - public double PM25_24HAQI { get; set; } - public int battery { get; set; } + public int? channel { get; set; } + public double? PM25 { get; set; } + public double? PM25_RealAQI { get; set; } + public double? PM25_24HAQI { get; set; } + public int? battery { get; set; } } public class ch_leakSensor { - public int channel { get; set; } + public int? channel { get; set; } public string name { get; set; } - public int battery { get; set; } + public int? battery { get; set; } public string status { get; set; } } @@ -384,8 +387,5 @@ public class liveData public tempHumSensor[]? ch_temp { get; set; } public tempHumSensor[]? ch_leaf { get; set; } } - - } - } diff --git a/CumulusMX/StationSettings.cs b/CumulusMX/StationSettings.cs index 1342e4c0..1929995c 100644 --- a/CumulusMX/StationSettings.cs +++ b/CumulusMX/StationSettings.cs @@ -157,6 +157,10 @@ internal string GetAlpacaFormData() macaddress = cumulus.Gw1000MacAddress, }; + var ecowittHttpApi = new JsonStationSettingsHttpApi() + { + ipaddress = cumulus.Gw1000IpAddress + }; var ecowitt = new JsonStationSettingsEcowitt { @@ -441,6 +445,7 @@ internal string GetAlpacaFormData() davisvp2 = davisvp2, daviswll = wll, gw1000 = gw1000, + ecowitthttpapi = ecowittHttpApi, ecowitt = ecowitt, ecowittapi = ecowittapi, ecowittmaps = ecowittmaps, @@ -906,6 +911,22 @@ internal string UpdateConfig(IHttpContext context) context.Response.StatusCode = 500; } + // HTTP Local API connection details + try + { + if (settings.ecowitthttpapi != null) + { + cumulus.Gw1000IpAddress = string.IsNullOrWhiteSpace(settings.ecowitthttpapi.ipaddress) ? null : settings.ecowitthttpapi.ipaddress.Trim(); + } + } + catch (Exception ex) + { + var msg = "Error processing GW1000 settings: " + ex.Message; + cumulus.LogErrorMessage(msg); + errorMsg += msg + "\n\n"; + context.Response.StatusCode = 500; + } + // Ecowitt configuration details try { @@ -1605,6 +1626,7 @@ internal class JsonStationSettingsData public JsonStationGeneral general { get; set; } public JsonStationSettingsDavisVp2 davisvp2 { get; set; } public JsonStationSettingsGw1000Conn gw1000 { get; set; } + public JsonStationSettingsHttpApi ecowitthttpapi { get; set; } public JsonStationSettingsEcowitt ecowitt { get; set; } public JsonStationSettingsEcowittApi ecowittapi { get; set; } public JsonStationSettingsEcowittMappings ecowittmaps { get; set; } @@ -1780,6 +1802,11 @@ internal class JsonStationSettingsGw1000Conn public int primaryRainSensor { get; set; } } + internal class JsonStationSettingsHttpApi + { + public string ipaddress { get; set; } + } + internal class JsonStationSettingsEcowitt { public bool setcustom { get; set; } diff --git a/CumulusMX/Wizard.cs b/CumulusMX/Wizard.cs index 4c960e25..fbf08716 100644 --- a/CumulusMX/Wizard.cs +++ b/CumulusMX/Wizard.cs @@ -110,6 +110,11 @@ public string GetAlpacaFormData() macaddress = cumulus.Gw1000MacAddress }; + var ecowittHttpApi = new JsonStationSettingsHttpApi() + { + ipaddress = cumulus.Gw1000IpAddress + }; + var fineoffset = new JsonWizardFineOffset() { syncreads = cumulus.FineOffsetOptions.SyncReads, @@ -830,6 +835,7 @@ internal class JsonWizardStation public JsonWizardDavisWll daviswll { get; set; } public JsonWizardDavisWll daviscloud { get; set; } public JsonStationSettingsGw1000Conn gw1000 { get; set; } + public JsonStationSettingsHttpApi ecowitthttpapi { get; set; } public JsonWizardFineOffset fineoffset { get; set; } public JsonWizardEasyWeather easyw { get; set; } public JsonWizardImet imet { get; set; } diff --git a/Updates.txt b/Updates.txt index 3f04ce4c..c495887e 100644 --- a/Updates.txt +++ b/Updates.txt @@ -1,4 +1,4 @@ -4.2.0 - b4030 +4.2.0 - b4031 ————————————— New - Exposes the UseDataLogger setting in Station Settings > Options > Advanced @@ -6,6 +6,7 @@ New - Implements a new data viewer where you can select and view historic daily data from the day file for a given period, for a set of data values. See: "Data logs > Daily Data Viewer" Changed +- The Station Settings screen now does a two stage selection: First the manufacturer, then the station model. This shortens the long and list to select from. Fixed From 6c712325ede75024b120c05ab49dffa03c40170c Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sun, 1 Sep 2024 22:55:13 +0100 Subject: [PATCH 16/63] Apply manufacturer/station changes to the config Wizard --- CumulusMX/Wizard.cs | 3 +++ 1 file changed, 3 insertions(+) diff --git a/CumulusMX/Wizard.cs b/CumulusMX/Wizard.cs index fbf08716..b70bff7e 100644 --- a/CumulusMX/Wizard.cs +++ b/CumulusMX/Wizard.cs @@ -159,6 +159,7 @@ public string GetAlpacaFormData() var station = new JsonWizardStation() { + manufacturer = cumulus.Manufacturer, stationtype = cumulus.StationType, stationmodel = cumulus.StationModel, davisvp2 = davisvp, @@ -509,6 +510,7 @@ public string UpdateConfig(IHttpContext context) cumulus.LogWarningMessage("Station type changed, restart required"); Cumulus.LogConsoleMessage("*** Station type changed, restart required ***", ConsoleColor.Yellow); } + cumulus.Manufacturer = settings.station.manufacturer; cumulus.StationType = settings.station.stationtype; cumulus.StationModel = settings.station.stationmodel; } @@ -829,6 +831,7 @@ internal class JsonWizardLogs internal class JsonWizardStation { + public int manufacturer { get; set; } public int stationtype { get; set; } public string stationmodel { get; set; } public JsonWizardDavisVp davisvp2 { get; set; } From 7188b5228e9363836327738476c40fad7272273d Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sun, 1 Sep 2024 23:26:57 +0100 Subject: [PATCH 17/63] More Ecowitt HTTP Local API updates More station/wizard settings updates --- CumulusMX/EcowittHttpApiStation.cs | 133 ++++++++++++++++------------- CumulusMX/StationSettings.cs | 2 +- CumulusMX/Wizard.cs | 18 ++++ 3 files changed, 95 insertions(+), 58 deletions(-) diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index 2a4939da..4b28b465 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -733,78 +733,97 @@ private void ProcessRain(EcowittLocalApi.commonSensor[] sensors, DateTime dateTi { foreach (var sensor in sensors) { - //Rain Event (val unit) - try + switch (sensor.id) { - var arr = sensor.val.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out var val)) - { - var evnt = arr[1] switch + case "0x0D": + //Rain Event (val unit) + try { - "mm" => ConvertUnits.RainMMToUser(val), - "in" => ConvertUnits.RainINToUser(val), - _ => -999 - }; + var arr = sensor.val.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out var val)) + { + var evnt = arr[1] switch + { + "mm" => ConvertUnits.RainMMToUser(val), + "in" => ConvertUnits.RainINToUser(val), + _ => -999 + }; - if (evnt >= 0) + if (evnt >= 0) + { + StormRain = evnt; + } + } + } + catch (Exception ex) { - StormRain = evnt; + //TODO: log a message } - } - } - catch (Exception ex) - { - //TODO: log a message - } + break; - //Rain Rate (val unit/h) - try - { - var arr = sensor.val.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out var val)) - { - var rate = arr[1] switch + case "0x0E": + //Rain Rate (val unit/h) + try { - "mm/Hr" => ConvertUnits.RainMMToUser(val), - "in/Hr" => ConvertUnits.RainINToUser(val), - _ => -999 - }; + var arr = sensor.val.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out var val)) + { + var rate = arr[1] switch + { + "mm/Hr" => ConvertUnits.RainMMToUser(val), + "in/Hr" => ConvertUnits.RainINToUser(val), + _ => -999 + }; - if (rate >= 0) + if (rate >= 0) + { + rainRateLast = rate; + } + } + } + catch (Exception ex) { - rainRateLast = rate; + //TODO: log a message } - } - } - catch (Exception ex) - { - //TODO: log a message - } - - //Rain Year (val unit) - try - { - // TODO: battery status + break; - var arr = sensor.val.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out var val)) - { - var yr = arr[1] switch + case "0x13": + //Rain Year (val unit) + try { - "mm/Hr" => ConvertUnits.RainMMToUser(val), - "in/Hr" => ConvertUnits.RainINToUser(val), - _ => -999 - }; + // TODO: battery status - if (yr >= 0) + var arr = sensor.val.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out var val)) + { + var yr = arr[1] switch + { + "mm" => ConvertUnits.RainMMToUser(val), + "in" => ConvertUnits.RainINToUser(val), + _ => -999 + }; + + if (yr >= 0) + { + rainLast = yr; + } + } + } + catch (Exception ex) { - rainLast = yr; + //TODO: log a message } - } - } - catch (Exception ex) - { - //TODO: log a message + break; + + case "0x10": // Rain day + case "0x11": // Rain week + case "0x12": // Rain month + // do nothing + break; + + default: + cumulus.LogDebugMessage($"Error: Unknown rain sensor id found = {sensor.id}"); + break; } } } diff --git a/CumulusMX/StationSettings.cs b/CumulusMX/StationSettings.cs index 1929995c..7b702b0e 100644 --- a/CumulusMX/StationSettings.cs +++ b/CumulusMX/StationSettings.cs @@ -921,7 +921,7 @@ internal string UpdateConfig(IHttpContext context) } catch (Exception ex) { - var msg = "Error processing GW1000 settings: " + ex.Message; + var msg = "Error processing Ecowitt Local HTTP API settings: " + ex.Message; cumulus.LogErrorMessage(msg); errorMsg += msg + "\n\n"; context.Response.StatusCode = 500; diff --git a/CumulusMX/Wizard.cs b/CumulusMX/Wizard.cs index b70bff7e..aefee8a8 100644 --- a/CumulusMX/Wizard.cs +++ b/CumulusMX/Wizard.cs @@ -166,6 +166,7 @@ public string GetAlpacaFormData() daviswll = daviswll, daviscloud = daviscloud, gw1000 = gw1000, + ecowitthttpapi = ecowittHttpApi, fineoffset = fineoffset, easyw = easyweather, imet = imet, @@ -629,6 +630,23 @@ public string UpdateConfig(IHttpContext context) context.Response.StatusCode = 500; } + + // HTTP Local API connection details + try + { + if (settings.station.ecowitthttpapi != null) + { + cumulus.Gw1000IpAddress = string.IsNullOrWhiteSpace(settings.station.ecowitthttpapi.ipaddress) ? null : settings.station.ecowitthttpapi.ipaddress.Trim(); + } + } + catch (Exception ex) + { + var msg = "Error processing Ecowitt Local HTTP API settings: " + ex.Message; + cumulus.LogErrorMessage(msg); + errorMsg += msg + "\n\n"; + context.Response.StatusCode = 500; + } + // weatherflow connection details try { From 3a53f18781c2054372906c0927b93e043952d947 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Mon, 2 Sep 2024 11:35:29 +0100 Subject: [PATCH 18/63] Ecowitt Local HTTP API error handling added --- CumulusMX/EcowittHttpApiStation.cs | 526 +++++++++++++++++------------ 1 file changed, 308 insertions(+), 218 deletions(-) diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index 4b28b465..a4257237 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -512,123 +512,132 @@ private void ProcessCommonList(EcowittLocalApi.commonSensor[] sensors, DateTime { cumulus.LogDebugMessage($"ProcessCommonList: Processing {sensors.Length} sensors"); - foreach (var sensor in sensors) + for (var i = 0; i < sensors.Length; i++) { - switch (sensor.id) - { - case "0x02": //Outdoor Temperature - if (sensor.valDbl.HasValue && cumulus.Gw1000PrimaryTHSensor == 0) - { - // do not process temperature here as if "MX calculates DP" is enabled, we have not yet read the humidity value. Have to do it at the end. - outdoortemp = sensor.valDbl.Value; - outdoortemp = sensor.unit == "C" ? ConvertUnits.TempCToUser(outdoortemp) : ConvertUnits.TempFToUser(outdoortemp); - } - break; - case "0x03": //Dew point - if (sensor.valDbl.HasValue && cumulus.Gw1000PrimaryTHSensor == 0 && !cumulus.StationOptions.CalculatedDP) - { - var temp = sensor.valDbl.Value; - temp = sensor.unit == "C" ? ConvertUnits.TempCToUser(temp) : ConvertUnits.TempFToUser(temp); + var sensor = sensors[i]; - DoOutdoorDewpoint(temp, dateTime); - } - break; - case "3": //Feels like - // do nothing with this for now - MX calcuates feels like - break; - case "0x04": //Wind chill - if (sensor.valDbl.HasValue && cumulus.Gw1000PrimaryTHSensor == 0) - { - windchill = sensor.valDbl.Value; - windchill = sensor.unit == "C" ? ConvertUnits.TempCToUser(windchill) : ConvertUnits.TempFToUser(windchill); - } - break; - case "0x05": //Heat index - // cumulus calculates this - break; - case "0x07": //Outdoor Humidity (%) - if (sensor.valInt.HasValue && cumulus.Gw1000PrimaryTHSensor == 0) - { - DoOutdoorHumidity(sensor.valInt.Value, dateTime); - } - break; - case "0x0A": //Wind Direction (360°) - if (sensor.valInt.HasValue) - { - windDirLast = sensor.valInt.Value; - } - break; - case "0x0B": //Wind Speed (val unit) - var arr = sensor.val.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out var valDbl)) - { - var spd = arr[1] switch + try + { + switch (sensor.id) + { + case "0x02": //Outdoor Temperature + if (sensor.valDbl.HasValue && cumulus.Gw1000PrimaryTHSensor == 0) { - "km/h" => ConvertUnits.WindKPHToUser(valDbl), - "m/s" => ConvertUnits.WindMSToUser(valDbl), - "mph" => ConvertUnits.WindMPHToUser(valDbl), - "knots" => ConvertUnits.WindKnotsToUser(valDbl), - _ => -999 - }; + // do not process temperature here as if "MX calculates DP" is enabled, we have not yet read the humidity value. Have to do it at the end. + outdoortemp = sensor.valDbl.Value; + outdoortemp = sensor.unit == "C" ? ConvertUnits.TempCToUser(outdoortemp) : ConvertUnits.TempFToUser(outdoortemp); + } + break; + case "0x03": //Dew point + if (sensor.valDbl.HasValue && cumulus.Gw1000PrimaryTHSensor == 0 && !cumulus.StationOptions.CalculatedDP) + { + var temp = sensor.valDbl.Value; + temp = sensor.unit == "C" ? ConvertUnits.TempCToUser(temp) : ConvertUnits.TempFToUser(temp); - if (spd >= 0) + DoOutdoorDewpoint(temp, dateTime); + } + break; + case "3": //Feels like + // do nothing with this for now - MX calcuates feels like + break; + case "0x04": //Wind chill + if (sensor.valDbl.HasValue && cumulus.Gw1000PrimaryTHSensor == 0) { - windSpeedLast = spd; + windchill = sensor.valDbl.Value; + windchill = sensor.unit == "C" ? ConvertUnits.TempCToUser(windchill) : ConvertUnits.TempFToUser(windchill); } - } - break; - case "0x0C": // Gust speed (val unit) - arr = sensor.val.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out valDbl)) - { - var spd = arr[1] switch + break; + case "0x05": //Heat index + // cumulus calculates this + break; + case "0x07": //Outdoor Humidity (%) + if (sensor.valInt.HasValue && cumulus.Gw1000PrimaryTHSensor == 0) { - "km/h" => ConvertUnits.WindKPHToUser(valDbl), - "m/s" => ConvertUnits.WindMSToUser(valDbl), - "mph" => ConvertUnits.WindMPHToUser(valDbl), - "knots" => ConvertUnits.WindKnotsToUser(valDbl), - _ => -999 - }; + DoOutdoorHumidity(sensor.valInt.Value, dateTime); + } + break; + case "0x0A": //Wind Direction (360°) + if (sensor.valInt.HasValue) + { + windDirLast = sensor.valInt.Value; + } + break; + case "0x0B": //Wind Speed (val unit) + var arr = sensor.val.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out var valDbl)) + { + var spd = arr[1] switch + { + "km/h" => ConvertUnits.WindKPHToUser(valDbl), + "m/s" => ConvertUnits.WindMSToUser(valDbl), + "mph" => ConvertUnits.WindMPHToUser(valDbl), + "knots" => ConvertUnits.WindKnotsToUser(valDbl), + _ => -999 + }; - if (spd >= 0) + if (spd >= 0) + { + windSpeedLast = spd; + } + } + break; + case "0x0C": // Gust speed (val unit) + arr = sensor.val.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out valDbl)) { - gustLast = spd; + var spd = arr[1] switch + { + "km/h" => ConvertUnits.WindKPHToUser(valDbl), + "m/s" => ConvertUnits.WindMSToUser(valDbl), + "mph" => ConvertUnits.WindMPHToUser(valDbl), + "knots" => ConvertUnits.WindKnotsToUser(valDbl), + _ => -999 + }; + + if (spd >= 0) + { + gustLast = spd; + } } - } - break; - case "0x15": //Light (value unit) - arr = sensor.val.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out valDbl)) - { - var light = arr[1] switch + break; + case "0x15": //Light (value unit) + arr = sensor.val.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out valDbl)) { - "fc" => valDbl * 0.015759751708199, - "lux" => valDbl * cumulus.SolarOptions.LuxToWM2, // convert Lux to W/m² - approximately! - "W/m2" => valDbl, - _ => -999 - }; + var light = arr[1] switch + { + "fc" => valDbl * 0.015759751708199, + "lux" => valDbl * cumulus.SolarOptions.LuxToWM2, // convert Lux to W/m² - approximately! + "W/m2" => valDbl, + _ => -999 + }; - LightValue = valDbl; - if (light >= 0) + LightValue = valDbl; + if (light >= 0) + { + DoSolarRad((int) light, dateTime); + } + } + break; + case "0x17": //UVI (0-15 index) + if (sensor.valDbl.HasValue) { - DoSolarRad((int) light, dateTime); + DoUV(sensor.valDbl.Value, dateTime); } - } - break; - case "0x17": //UVI (0-15 index) - if (sensor.valDbl.HasValue) - { - DoUV(sensor.valDbl.Value, dateTime); - } - break; + break; - case "0x19": // max wind today (value unit) - // not used - break; + case "0x19": // max wind today (value unit) + // not used + break; - default: - cumulus.LogDebugMessage($"Error: Unknown common_list sensor id found = {sensor.id}"); - break; + default: + cumulus.LogDebugMessage($"Error: Unknown common_list sensor id found = {sensor.id}"); + break; + } + } + catch(Exception ex) + { + cumulus.LogExceptionMessage(ex, $"ProcessCommonList: Error processing sensor id {sensor.id}"); } } @@ -639,8 +648,10 @@ private void ProcessCommonList(EcowittLocalApi.commonSensor[] sensors, DateTime private void ProcessWh25(EcowittLocalApi.wh25Sensor[] sensors, DateTime dateTime) { - foreach (var sensor in sensors) + for (var i = 0; i < sensors.Length; i++) { + var sensor = sensors[i]; + // Indoor Temperature try { @@ -657,7 +668,7 @@ private void ProcessWh25(EcowittLocalApi.wh25Sensor[] sensors, DateTime dateTime } catch (Exception ex) { - //TODO: log a message + cumulus.LogExceptionMessage(ex, "ProcessWh25: Error processing indoor temp}"); } // Indoor Humidity @@ -673,7 +684,7 @@ private void ProcessWh25(EcowittLocalApi.wh25Sensor[] sensors, DateTime dateTime } catch (Exception ex) { - //TODO: log a message + cumulus.LogExceptionMessage(ex, "ProcessWh25: Error processing indoor humidity}"); } // Pressure @@ -722,7 +733,7 @@ private void ProcessWh25(EcowittLocalApi.wh25Sensor[] sensors, DateTime dateTime } catch (Exception ex) { - //TODO: log a message + cumulus.LogExceptionMessage(ex, "ProcessWh25: Error processing pressure}"); } // TODO: battery status @@ -731,109 +742,121 @@ private void ProcessWh25(EcowittLocalApi.wh25Sensor[] sensors, DateTime dateTime private void ProcessRain(EcowittLocalApi.commonSensor[] sensors, DateTime dateTime) { - foreach (var sensor in sensors) + for (var i = 0; i < sensors.Length; i++) { - switch (sensor.id) + var sensor = sensors[i]; + + try { - case "0x0D": - //Rain Event (val unit) - try - { - var arr = sensor.val.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out var val)) + switch (sensor.id) + { + case "0x0D": + //Rain Event (val unit) + try { - var evnt = arr[1] switch + var arr = sensor.val.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out var val)) { - "mm" => ConvertUnits.RainMMToUser(val), - "in" => ConvertUnits.RainINToUser(val), - _ => -999 - }; + var evnt = arr[1] switch + { + "mm" => ConvertUnits.RainMMToUser(val), + "in" => ConvertUnits.RainINToUser(val), + _ => -999 + }; - if (evnt >= 0) - { - StormRain = evnt; + if (evnt >= 0) + { + StormRain = evnt; + } } } - } - catch (Exception ex) - { - //TODO: log a message - } - break; + catch (Exception ex) + { + //TODO: log a message + } + break; - case "0x0E": - //Rain Rate (val unit/h) - try - { - var arr = sensor.val.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out var val)) + case "0x0E": + //Rain Rate (val unit/h) + try { - var rate = arr[1] switch + var arr = sensor.val.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out var val)) { - "mm/Hr" => ConvertUnits.RainMMToUser(val), - "in/Hr" => ConvertUnits.RainINToUser(val), - _ => -999 - }; + var rate = arr[1] switch + { + "mm/Hr" => ConvertUnits.RainMMToUser(val), + "in/Hr" => ConvertUnits.RainINToUser(val), + _ => -999 + }; - if (rate >= 0) - { - rainRateLast = rate; + if (rate >= 0) + { + rainRateLast = rate; + } } } - } - catch (Exception ex) - { - //TODO: log a message - } - break; - - case "0x13": - //Rain Year (val unit) - try - { - // TODO: battery status + catch (Exception ex) + { + //TODO: log a message + } + break; - var arr = sensor.val.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out var val)) + case "0x13": + //Rain Year (val unit) + try { - var yr = arr[1] switch - { - "mm" => ConvertUnits.RainMMToUser(val), - "in" => ConvertUnits.RainINToUser(val), - _ => -999 - }; + // TODO: battery status - if (yr >= 0) + var arr = sensor.val.Split(' '); + if (arr.Length == 2 && double.TryParse(arr[0], out var val)) { - rainLast = yr; + var yr = arr[1] switch + { + "mm" => ConvertUnits.RainMMToUser(val), + "in" => ConvertUnits.RainINToUser(val), + _ => -999 + }; + + if (yr >= 0) + { + rainLast = yr; + } } } - } - catch (Exception ex) - { - //TODO: log a message - } - break; + catch (Exception ex) + { + //TODO: log a message + } + break; - case "0x10": // Rain day - case "0x11": // Rain week - case "0x12": // Rain month - // do nothing - break; + case "0x10": // Rain day + case "0x11": // Rain week + case "0x12": // Rain month + // do nothing + break; - default: - cumulus.LogDebugMessage($"Error: Unknown rain sensor id found = {sensor.id}"); - break; + default: + cumulus.LogDebugMessage($"Error: Unknown rain sensor id found = {sensor.id}"); + break; + } } + catch (Exception ex) + { + cumulus.LogExceptionMessage(ex, $"ProcessRain: Error processing sensor id {sensor.id}"); + } + } } private void ProcessLightning(EcowittLocalApi.lightningSensor[] sensors, DateTime dateTime) { - foreach (var sensor in sensors) + for (var i = 0; i < sensors.Length; i++) { try { + var sensor = sensors[i]; + //Lightning dist (1-40km) if (sensor.distanceVal.HasValue && sensor.distanceUnit != null) { @@ -888,7 +911,7 @@ private void ProcessLightning(EcowittLocalApi.lightningSensor[] sensors, DateTim } catch (Exception ex) { - //TODO: log a message + cumulus.LogExceptionMessage(ex, "ProcessLightning: Error"); } } } @@ -897,10 +920,12 @@ private void ProcessCo2(EcowittLocalApi.co2Sensor[] sensors, DateTime dateTime) { cumulus.LogDebugMessage("WH45 CO₂: Decoding..."); - foreach (var sensor in sensors) + for (var i = 0; i < sensors.Length; i++) { try { + var sensor = sensors[i]; + if (sensor.temp.HasValue && !string.IsNullOrEmpty(sensor.unit)) { CO2_temperature = sensor.unit == "C" ? ConvertUnits.TempCToUser(sensor.temp.Value) : ConvertUnits.TempFToUser(sensor.temp.Value); @@ -942,11 +967,20 @@ private void ProcessCo2(EcowittLocalApi.co2Sensor[] sensors, DateTime dateTime) private void ProcessChPm25(EcowittLocalApi.ch_pm25Sensor[] sensors) { - foreach (var sensor in sensors) + for (var i = 0; i < sensors.Length; i++) { - if (sensor.channel.HasValue && sensor.PM25.HasValue) + var sensor = sensors[i]; + + try { - DoAirQuality(sensor.PM25.Value, sensor.channel.Value); + if (sensor.channel.HasValue && sensor.PM25.HasValue) + { + DoAirQuality(sensor.PM25.Value, sensor.channel.Value); + } + } + catch (Exception ex) + { + cumulus.LogExceptionMessage(ex, "ProcessChPm25: Error"); } // TODO: Battery status @@ -955,35 +989,54 @@ private void ProcessChPm25(EcowittLocalApi.ch_pm25Sensor[] sensors) private void ProcessLeak(EcowittLocalApi.ch_leakSensor[] sensors) { - foreach (var sensor in sensors) + for (var i = 0; i < sensors.Length; i++) { - if (sensor.channel.HasValue && !string.IsNullOrEmpty(sensor.status)) + var sensor = sensors[i]; + + try { - var val = sensor.status == "NORMAL" ? 0 : 1; - DoLeakSensor(val, sensor.channel.Value); + if (sensor.channel.HasValue && !string.IsNullOrEmpty(sensor.status)) + { + var val = sensor.status == "NORMAL" ? 0 : 1; + DoLeakSensor(val, sensor.channel.Value); - // TODO: Battery status + // TODO: Battery status + } + } + catch (Exception ex) + { + cumulus.LogExceptionMessage(ex, $"ProcessLeak: Error processing channel {sensor.channel.Value}"); } + } } private void ProcessExtraTempHum(EcowittLocalApi.tempHumSensor[] sensors, DateTime dateTime) { - foreach (var sensor in sensors) + for (var i = 0; i < sensors.Length; i++) { - if (sensor.temp.HasValue) - { - DoExtraTemp(sensor.temp.Value, sensor.channel); - } + var sensor = sensors[i]; - if (sensor.humidityVal.HasValue) + try { - if (cumulus.Gw1000PrimaryTHSensor == sensor.channel) + if (sensor.temp.HasValue) { - DoOutdoorHumidity(sensor.humidityVal.Value, dateTime); + DoExtraTemp(sensor.temp.Value, sensor.channel); } - DoExtraHum(sensor.humidityVal.Value, sensor.channel); + + if (sensor.humidityVal.HasValue) + { + if (cumulus.Gw1000PrimaryTHSensor == sensor.channel) + { + DoOutdoorHumidity(sensor.humidityVal.Value, dateTime); + } + DoExtraHum(sensor.humidityVal.Value, sensor.channel); + } + } + catch (Exception ex) + { + cumulus.LogExceptionMessage(ex, $"ProcessExtraTempHum: Error processind sensor channel {sensor.channel}"); } // TODO: battery status @@ -994,32 +1047,50 @@ private void ProcessUserTemp(EcowittLocalApi.tempHumSensor[] sensors, DateTime d { // user temp = WH34 8 channel Soil or Water temperature sensors - foreach (var sensor in sensors) + for (var i = 0; i < sensors.Length; i++) { - if (sensor.temp.HasValue) + var sensor = sensors[i]; + + try { - var val = sensor.unit == "C" ? ConvertUnits.TempCToUser(sensor.temp.Value) : ConvertUnits.TempFToUser(sensor.temp.Value); - if (cumulus.EcowittMapWN34[sensor.channel] == 0) // false = user temp, true = soil temp - { - DoUserTemp(val, sensor.channel); - } - else + if (sensor.temp.HasValue) { - DoSoilTemp(val, cumulus.EcowittMapWN34[sensor.channel]); + var val = sensor.unit == "C" ? ConvertUnits.TempCToUser(sensor.temp.Value) : ConvertUnits.TempFToUser(sensor.temp.Value); + if (cumulus.EcowittMapWN34[sensor.channel] == 0) // false = user temp, true = soil temp + { + DoUserTemp(val, sensor.channel); + } + else + { + DoSoilTemp(val, cumulus.EcowittMapWN34[sensor.channel]); + } } - - // TODO: Battery status } + catch (Exception ex) + { + cumulus.LogExceptionMessage(ex, $"ProcessUserTemp: Error processing sensor channel {sensor.channel}"); + } + + // TODO: Battery status } } private void ProcessSoilMoisture(EcowittLocalApi.tempHumSensor[] sensors) { - foreach (var sensor in sensors) + for (var i = 0; i < sensors.Length; i++) { - if (sensor.humidityVal.HasValue) + try + { + var sensor = sensors[i]; + + if (sensor.humidityVal.HasValue) + { + DoSoilMoisture(sensor.humidityVal.Value, sensor.channel); + } + } + catch (Exception ex) { - DoSoilMoisture(sensor.humidityVal.Value, sensor.channel); + cumulus.LogExceptionMessage(ex, "ProcessSoilMoisture: Error"); } // TODO: Battery status @@ -1029,12 +1100,22 @@ private void ProcessSoilMoisture(EcowittLocalApi.tempHumSensor[] sensors) private void ProcessSoilTemp(EcowittLocalApi.tempHumSensor[] sensors) { - foreach (var sensor in sensors) + for (var i = 0; i < sensors.Length; i++) { - if (sensor.temp.HasValue) + var sensor = sensors[i]; + + try + { + + if (sensor.temp.HasValue) + { + var val = sensor.unit == "C" ? ConvertUnits.TempCToUser(sensor.temp.Value) : ConvertUnits.TempFToUser(sensor.temp.Value); + DoSoilTemp(val, sensor.channel); + } + } + catch (Exception ex) { - var val = sensor.unit == "C" ? ConvertUnits.TempCToUser(sensor.temp.Value) : ConvertUnits.TempFToUser(sensor.temp.Value); - DoSoilTemp(val, sensor.channel); + cumulus.LogExceptionMessage(ex, $"ProcessSoilTemp: Error processing channel {sensor.channel}"); } // TODO: Battery status @@ -1044,11 +1125,20 @@ private void ProcessSoilTemp(EcowittLocalApi.tempHumSensor[] sensors) private void ProcessLeafWet(EcowittLocalApi.tempHumSensor[] sensors) { - foreach (var sensor in sensors) + for (var i = 0; i < sensors.Length; i++) { - if (sensor.humidityVal.HasValue) + var sensor = sensors[i]; + + try + { + if (sensor.humidityVal.HasValue) + { + DoLeafWetness(sensor.humidityVal.Value, sensor.channel); + } + } + catch (Exception ex) { - DoLeafWetness(sensor.humidityVal.Value, sensor.channel); + cumulus.LogExceptionMessage(ex, $"ProcessLeafWet: Error processing channel {sensor.channel}"); } // TODO: Battery status From 63df46152bab4dc82a7733c868228fc6d46dd2ed Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Mon, 2 Sep 2024 22:57:46 +0100 Subject: [PATCH 19/63] Add battery state detection to Ecowitt Local HTTP API Add new web tag <#LowBatteryList> --- CumulusMX/EcowittHttpApiStation.cs | 281 ++++++++++++++------------- CumulusMX/EcowittLocalApi.cs | 297 ++++++++++++++++++++++++----- CumulusMX/WeatherStation.cs | 2 + CumulusMX/webtags.cs | 6 + Updates.txt | 8 + 5 files changed, 414 insertions(+), 180 deletions(-) diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index a4257237..5806b0d3 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -15,11 +15,11 @@ namespace CumulusMX internal class EcowittHttpApiStation : WeatherStation #pragma warning restore CA1001 // Types that own disposable fields should be disposable { - private string ipaddr; + private readonly string ipaddr; private readonly string macaddr; private string deviceModel; private string deviceFirmware; - private int updateRate = 10000; // 10 seconds by default + private readonly int updateRate = 10000; // 10 seconds by default private int lastMinute = -1; private int lastHour = -1; @@ -145,6 +145,9 @@ public override void Start() cumulus.StartTimersAndSensors(); + // Get the sensor list + GetSensorIds().Wait(); + liveTask = Task.Run(() => { try @@ -166,13 +169,11 @@ public override void Start() // process rain values if (cumulus.Gw1000PrimaryRainSensor == 0 && rawData.rain != null) { - ProcessRain(rawData.rain, dataLastRead); - // TODO: battery status for piezo + ProcessRain(rawData.rain); } else if (cumulus.Gw1000PrimaryRainSensor == 1 && rawData.piezoRain != null) { - ProcessRain(rawData.piezoRain, dataLastRead); - // TODO: battery status for tipper + ProcessRain(rawData.piezoRain); } if (rawData.lightning != null) @@ -202,7 +203,7 @@ public override void Start() if (rawData.ch_temp != null) { - ProcessUserTemp(rawData.ch_temp, dataLastRead); + ProcessUserTemp(rawData.ch_temp); } if (rawData.ch_soil != null) @@ -210,12 +211,6 @@ public override void Start() ProcessSoilMoisture(rawData.ch_soil); } - // TODO: Soil Temperature sensors - //if (rawData.ch_ ??? != null) - //{ - // ProcessSoilTemp(rawData.ch_ ???); - //} - if (rawData.ch_leaf != null) { ProcessLeafWet(rawData.ch_leaf); @@ -287,6 +282,12 @@ public override void Start() { lastMinute = minute; + // at the start of every 20 minutes to trigger battery status check + if ((minute % 20) == 0 && !cumulus.cancellationToken.IsCancellationRequested) + { + _ = GetSensorIds(); + } + // every day dump the clock drift at midday each day if (minute == 0 && DateTime.Now.Hour == 12) { @@ -495,20 +496,110 @@ private async Task CheckAvailableFirmware() } } - private void GetSensorIdsNew() + private async Task GetSensorIds() { cumulus.LogMessage("Reading sensor ids"); - } + var sensors = await localApi.GetSensorInfo(cumulus.cancellationToken); + batteryLow = false; + bool sensorOk; - private bool PrintSensorInfoNew(byte[] data, int idx) - { - var batteryLow = false; + LowBatteryDevices.Clear(); + + if (sensors.Length > 0) + { + for (var i = 0; i< sensors.Length; i++) + { + var sensor = sensors[i]; + var name = string.Empty; + + cumulus.LogDebugMessage($" - enabled={sensor.idst}, type={sensor.img}, sensor id={sensor.id}, signal={sensor.signal}, battery={sensor.batt}, name={sensor.name}"); + + // check the battery status + if (sensor.idst && sensor.signal > 0) + { +#pragma warning disable S907 + switch (sensor.type) + { + case 0: // wh69 + break; + case 1: // wh68 + name = "wh68"; + goto case 1003; + case 2: // wh80 + name = "wh80"; + goto case 1003; + case 3: // wh40 + name = "wh40"; + goto case 1003; + case 4: // wh25 + name = "wh25"; + goto case 1003; + case 5: // wh26 + name = "wh326"; + goto case 1001; + case int n when (n > 5 && n < 14): // wh31 - T&H (8 chan) + name = "wh31ch" + (sensor.type - 5); + goto case 1001; + case int n when (n > 13 && n < 22): // wh51 - soil moisture (8 chan) + break; + case int n when (n > 21 && n < 26): // wh41 - pm2.5 (4 chan) + name = "wh41ch" + (sensor.type - 21); + goto case 1003; + case 26: // wh57 - lightning + name = "wh57"; + goto case 1003; + case int n when (n > 26 && n < 31): // wh55 - leak (4 chan) + name = "wh55ch" + (sensor.type - 26); + goto case 1003; + case int n when (n > 30 && n < 39): // wh34 - Temp (8 chan) + name = "wh34ch" + (sensor.type - 30); + goto case 1003; + case 39: // wh45 - co2 + name = "wh45"; + goto case 1003; + case int n when (n > 39 && n < 48): // wh35 - leasf wet (8 chan) + name = "wh35ch" + (sensor.type - 39); + goto case 1003; + case 48: // wh90 + name = "wh90"; + goto case 1003; + case 49: // wh85 + name = "wh85"; + goto case 1003; + + + case 1001: // battery type 1 (0=OK, 1=LOW) + if (sensor.batt == 1) + { + batteryLow = true; + LowBatteryDevices.Add(name + " - LOW"); + } + break; + + case 1003: // battery type 3 (0-1=LOW, 2-5=OK, 6=DC, 9=OFF) + if (!TestBattery3(sensor.batt)) + { + batteryLow = true; + LowBatteryDevices.Add(name + " - " + sensor.batt); + } + break; - return batteryLow; + default: + cumulus.LogWarningMessage($"Unknown sensor type in SendorIds. Model={sensor.img}, type={sensor.type}"); + break; + } +#pragma warning restore S907 + } + } + } + else + { + cumulus.LogWarningMessage("GetSensorIds: No sensor data returned"); + } } - private void ProcessCommonList(EcowittLocalApi.commonSensor[] sensors, DateTime dateTime) + private void ProcessCommonList(EcowittLocalApi.CommonSensor[] sensors, DateTime dateTime) { cumulus.LogDebugMessage($"ProcessCommonList: Processing {sensors.Length} sensors"); @@ -646,7 +737,7 @@ private void ProcessCommonList(EcowittLocalApi.commonSensor[] sensors, DateTime } - private void ProcessWh25(EcowittLocalApi.wh25Sensor[] sensors, DateTime dateTime) + private void ProcessWh25(EcowittLocalApi.Wh25Sensor[] sensors, DateTime dateTime) { for (var i = 0; i < sensors.Length; i++) { @@ -735,12 +826,10 @@ private void ProcessWh25(EcowittLocalApi.wh25Sensor[] sensors, DateTime dateTime { cumulus.LogExceptionMessage(ex, "ProcessWh25: Error processing pressure}"); } - - // TODO: battery status } } - private void ProcessRain(EcowittLocalApi.commonSensor[] sensors, DateTime dateTime) + private void ProcessRain(EcowittLocalApi.CommonSensor[] sensors) { for (var i = 0; i < sensors.Length; i++) { @@ -772,7 +861,7 @@ private void ProcessRain(EcowittLocalApi.commonSensor[] sensors, DateTime dateTi } catch (Exception ex) { - //TODO: log a message + cumulus.LogExceptionMessage(ex, $"ProcessRain: Error processing Rain Event for sensor id {sensor.id}"); } break; @@ -798,7 +887,7 @@ private void ProcessRain(EcowittLocalApi.commonSensor[] sensors, DateTime dateTi } catch (Exception ex) { - //TODO: log a message + cumulus.LogExceptionMessage(ex, $"ProcessRain: Error processing Rain Rate for sensor id {sensor.id}"); } break; @@ -806,8 +895,6 @@ private void ProcessRain(EcowittLocalApi.commonSensor[] sensors, DateTime dateTi //Rain Year (val unit) try { - // TODO: battery status - var arr = sensor.val.Split(' '); if (arr.Length == 2 && double.TryParse(arr[0], out var val)) { @@ -826,7 +913,7 @@ private void ProcessRain(EcowittLocalApi.commonSensor[] sensors, DateTime dateTi } catch (Exception ex) { - //TODO: log a message + cumulus.LogExceptionMessage(ex, $"ProcessRain: Error processing Rain Year for sensor id {sensor.id}"); } break; @@ -849,7 +936,7 @@ private void ProcessRain(EcowittLocalApi.commonSensor[] sensors, DateTime dateTi } } - private void ProcessLightning(EcowittLocalApi.lightningSensor[] sensors, DateTime dateTime) + private void ProcessLightning(EcowittLocalApi.LightningSensor[] sensors, DateTime dateTime) { for (var i = 0; i < sensors.Length; i++) { @@ -916,7 +1003,7 @@ private void ProcessLightning(EcowittLocalApi.lightningSensor[] sensors, DateTim } } - private void ProcessCo2(EcowittLocalApi.co2Sensor[] sensors, DateTime dateTime) + private void ProcessCo2(EcowittLocalApi.Co2Sensor[] sensors, DateTime dateTime) { cumulus.LogDebugMessage("WH45 CO₂: Decoding..."); @@ -952,8 +1039,6 @@ private void ProcessCo2(EcowittLocalApi.co2Sensor[] sensors, DateTime dateTime) { CO2_24h = sensor.CO2_24H.Value; } - - // TODO: Battery status } catch (Exception ex) { @@ -965,8 +1050,10 @@ private void ProcessCo2(EcowittLocalApi.co2Sensor[] sensors, DateTime dateTime) CO2_pm10_aqi = GetAqi(AqMeasure.pm10, CO2_pm10); } - private void ProcessChPm25(EcowittLocalApi.ch_pm25Sensor[] sensors) + private void ProcessChPm25(EcowittLocalApi.Ch_Pm25Sensor[] sensors) { + cumulus.LogDebugMessage($"ProcessChPm25: Processing {sensors.Length} sensors"); + for (var i = 0; i < sensors.Length; i++) { var sensor = sensors[i]; @@ -982,13 +1069,13 @@ private void ProcessChPm25(EcowittLocalApi.ch_pm25Sensor[] sensors) { cumulus.LogExceptionMessage(ex, "ProcessChPm25: Error"); } - - // TODO: Battery status } } - private void ProcessLeak(EcowittLocalApi.ch_leakSensor[] sensors) + private void ProcessLeak(EcowittLocalApi.Ch_LeakSensor[] sensors) { + cumulus.LogDebugMessage($"ProcessLeak: Processing {sensors.Length} sensors"); + for (var i = 0; i < sensors.Length; i++) { var sensor = sensors[i]; @@ -999,8 +1086,6 @@ private void ProcessLeak(EcowittLocalApi.ch_leakSensor[] sensors) { var val = sensor.status == "NORMAL" ? 0 : 1; DoLeakSensor(val, sensor.channel.Value); - - // TODO: Battery status } } catch (Exception ex) @@ -1011,9 +1096,10 @@ private void ProcessLeak(EcowittLocalApi.ch_leakSensor[] sensors) } } - - private void ProcessExtraTempHum(EcowittLocalApi.tempHumSensor[] sensors, DateTime dateTime) + private void ProcessExtraTempHum(EcowittLocalApi.TempHumSensor[] sensors, DateTime dateTime) { + cumulus.LogDebugMessage($"ProcessExtraTempHum: Processing {sensors.Length} sensors"); + for (var i = 0; i < sensors.Length; i++) { var sensor = sensors[i]; @@ -1038,14 +1124,13 @@ private void ProcessExtraTempHum(EcowittLocalApi.tempHumSensor[] sensors, DateTi { cumulus.LogExceptionMessage(ex, $"ProcessExtraTempHum: Error processind sensor channel {sensor.channel}"); } - - // TODO: battery status } } - private void ProcessUserTemp(EcowittLocalApi.tempHumSensor[] sensors, DateTime dateTime) + private void ProcessUserTemp(EcowittLocalApi.TempHumSensor[] sensors) { // user temp = WH34 8 channel Soil or Water temperature sensors + cumulus.LogDebugMessage($"ProcessUserTemp: Processing {sensors.Length} sensors"); for (var i = 0; i < sensors.Length; i++) { @@ -1070,13 +1155,13 @@ private void ProcessUserTemp(EcowittLocalApi.tempHumSensor[] sensors, DateTime d { cumulus.LogExceptionMessage(ex, $"ProcessUserTemp: Error processing sensor channel {sensor.channel}"); } - - // TODO: Battery status } } - private void ProcessSoilMoisture(EcowittLocalApi.tempHumSensor[] sensors) + private void ProcessSoilMoisture(EcowittLocalApi.TempHumSensor[] sensors) { + cumulus.LogDebugMessage($"ProcessSoilMoisture: Processing {sensors.Length} sensors"); + for (var i = 0; i < sensors.Length; i++) { try @@ -1092,39 +1177,13 @@ private void ProcessSoilMoisture(EcowittLocalApi.tempHumSensor[] sensors) { cumulus.LogExceptionMessage(ex, "ProcessSoilMoisture: Error"); } - - // TODO: Battery status } } - - private void ProcessSoilTemp(EcowittLocalApi.tempHumSensor[] sensors) + private void ProcessLeafWet(EcowittLocalApi.TempHumSensor[] sensors) { - for (var i = 0; i < sensors.Length; i++) - { - var sensor = sensors[i]; - - try - { + cumulus.LogDebugMessage($"ProcessLeafWet: Processing {sensors.Length} sensors"); - if (sensor.temp.HasValue) - { - var val = sensor.unit == "C" ? ConvertUnits.TempCToUser(sensor.temp.Value) : ConvertUnits.TempFToUser(sensor.temp.Value); - DoSoilTemp(val, sensor.channel); - } - } - catch (Exception ex) - { - cumulus.LogExceptionMessage(ex, $"ProcessSoilTemp: Error processing channel {sensor.channel}"); - } - - // TODO: Battery status - } - - } - - private void ProcessLeafWet(EcowittLocalApi.tempHumSensor[] sensors) - { for (var i = 0; i < sensors.Length; i++) { var sensor = sensors[i]; @@ -1140,72 +1199,38 @@ private void ProcessLeafWet(EcowittLocalApi.tempHumSensor[] sensors) { cumulus.LogExceptionMessage(ex, $"ProcessLeafWet: Error processing channel {sensor.channel}"); } - - // TODO: Battery status } } - private void GetSystemInfo(bool driftOnly) { - cumulus.LogMessage("Reading Ecowitt system info"); - - } + cumulus.LogMessage("NOT Reading Ecowitt system info"); - - private bool DoWH34BatteryStatus(byte[] data, int index) - { - // No longer used in firmware 1.6.0+ - cumulus.LogDebugMessage("WH34 battery status..."); - var str = "wh34>" + - " ch1=" + TestBattery3(data[index + 1]) + - " ch2=" + TestBattery3(data[index + 2]) + - " ch3=" + TestBattery3(data[index + 3]) + - " ch4=" + TestBattery3(data[index + 4]) + - " ch5=" + TestBattery3(data[index + 5]) + - " ch6=" + TestBattery3(data[index + 6]) + - " ch7=" + TestBattery3(data[index + 7]) + - " ch8=" + TestBattery3(data[index + 8]); - - cumulus.LogDebugMessage(str); - - return str.Contains("Low"); } - private static string TestBattery1(byte value, byte mask) + private static string TestBattery1(int value) { - return (value & mask) == 0 ? "OK" : "Low"; + return value == 0 ? "OK" : "Low"; } - private static string TestBattery3(byte value) + private static bool TestBattery3(int value) { if (value == 6) - return "DC"; + { + // DC + return true; + } if (value == 9) - return "OFF"; - return value > 1 ? "OK" : "Low"; - } - - private static string TestBattery10(byte value) - { - // consider 1.2V as low - if (value == 255) - return "n/a"; - - return value > 12 ? "OK" : "Low"; - } - - private static double TestBattery10V(byte value) - { - return value / 10.0; - } - - private static string TestBatteryWh40(byte value, double volts) - { - if (value == 255) - return "n/a"; + { + // OFF + return false; + } + if (value > 1) + { + return true; + } - return volts > 1.2 ? "OK" : "Low"; + return false; } private void DataTimeout(object source, ElapsedEventArgs e) @@ -1218,13 +1243,13 @@ private void DataTimeout(object source, ElapsedEventArgs e) } else { - cumulus.LogErrorMessage($"ERROR: No data received from the GW1000 for {tmrDataWatchdog.Interval / 1000} seconds"); + cumulus.LogErrorMessage($"ERROR: No data received from the sttaion for {tmrDataWatchdog.Interval / 1000} seconds"); if (!DataStopped) { DataStoppedTime = DateTime.Now; DataStopped = true; } - cumulus.DataStoppedAlarm.LastMessage = $"No data received from the GW1000 for {tmrDataWatchdog.Interval / 1000} seconds"; + cumulus.DataStoppedAlarm.LastMessage = $"No data received from the station for {tmrDataWatchdog.Interval / 1000} seconds"; cumulus.DataStoppedAlarm.Triggered = true; } } diff --git a/CumulusMX/EcowittLocalApi.cs b/CumulusMX/EcowittLocalApi.cs index 0fa005f3..cc77e2d1 100644 --- a/CumulusMX/EcowittLocalApi.cs +++ b/CumulusMX/EcowittLocalApi.cs @@ -1,5 +1,6 @@ using System; using System.Threading; +using System.Threading.Tasks; using ServiceStack; @@ -17,7 +18,7 @@ public EcowittLocalApi(Cumulus cumul) } - public liveData GetLiveData(CancellationToken token) + public LiveData GetLiveData(CancellationToken token) { // http://ip-address/get_livedata_info // @@ -180,7 +181,7 @@ public liveData GetLiveData(CancellationToken token) else if (responseBody.StartsWith('{')) // sanity check { // Convert JSON string to an object - liveData json = responseBody.FromJson(); + LiveData json = responseBody.FromJson(); return json; } } while (retries-- > 0); @@ -189,11 +190,221 @@ public liveData GetLiveData(CancellationToken token) } - public void GetSensorInfo() + public async Task GetSensorInfo(CancellationToken token) { // http://ip-address/get_sensors_info?page=1 // http://ip-address/get_sensors_info?page=2 + // Example of page=1 + /* + + [ + { + "img": "wh85", + "type": "49", + "name": "Wind & Rain", + "id": "FFFFFFFF", + "batt": "9", + "signal": "0", + "idst": "1" + }, + { + "img": "wh90", + "type": "48", + "name": "Temp & Humidity & Solar & Wind & Rain", + "id": "FFFFFFFE", + "batt": "9", + "signal": "0", + "idst": "0" + }, + { + "img": "wh69", + "type": "0", + "name": "Temp & Humidity & Solar & Wind & Rain", + "id": "FFFFFFFE", + "batt": "9", + "signal": "0", + "idst": "0" + }, + { + "img": "wh68", + "type": "1", + "name": "Solar & Wind", + "id": "FFFFFFFE", + "batt": "9", + "signal": "0", + "idst": "0" + }, + { + "img": "wh40", + "type": "3", + "name": "Rain", + "id": "E1EF", + "batt": "0", + "signal": "3", + "idst": "1" + }, + { + "img": "wh25", + "type": "4", + "name": "Temp & Humidity & Pressure", + "id": "FFFFFFFF", + "batt": "9", + "signal": "0", + "idst": "1" + }, + { + "img": "wh26", + "type": "5", + "name": "Temp & Humidity", + "id": "FFFFFFFE", + "batt": "9", + "signal": "0", + "idst": "0" + }, + { + "img": "wh80", + "type": "2", + "name": "Temp & Humidity & Solar & Wind", + "id": "E0038", + "batt": "5", + "signal": "4", + "idst": "1" + }, + { + "img": "wh57", + "type": "26", + "name": "Lightning", + "id": "FFFFFFFE", + "batt": "9", + "signal": "0", + "idst": "0" + }, + { + "img": "wh45", + "type": "39", + "name": "PM25 & PM10 & CO2", + "id": "FFFFFFFE", + "batt": "9", + "signal": "0", + "idst": "0" + }, + { + "img": "wh41", + "type": "22", + "name": "PM2.5 CH1", + "id": "34", + "batt": "9", + "signal": "0", + "idst": "1" + }, + .. chans 2-4 + { + "img": "wh55", + "type": "27", + "name": "Leak CH1", + "id": "FFFFFFFE", + "batt": "9", + "signal": "0", + "idst": "0" + }, + .. chans 2-4 + { + "img": "wh31", + "type": "6", + "name": "Temp & Humidity CH1", + "id": "B0", + "batt": "0", + "signal": "4", + "idst": "1" + }, + .. chans 2-7 + ] + */ + + + // Example page=2 + /* + [ + { + "img": "wh31", + "type": "13", + "name": "Temp & Humidity CH8", + "id": "FFFFFFFE", + "batt": "9", + "signal": "0", + "idst": "0" + }, + { + "img": "wh51", + "type": "14", + "name": "Soil moisture CH1", + "id": "FFFFFFFE", + "batt": "9", + "signal": "0", + "idst": "0" + }, + .. chans 2-8 + { + "img": "wh34", + "type": "31", + "name": "Temp CH1", + "id": "FFFFFFFE", + "batt": "9", + "signal": "0", + "idst": "0" + }, + .. chans 2-8 + { + "img": "wh35", + "type": "40", + "name": "Leaf Wetness CH1", + "id": "FFFFFFFE", + "batt": "9", + "signal": "0", + "idst": "0" + }, + .. chans 2-8 + ] + + */ + + string ip = cumulus.Gw1000IpAddress; + + SensorInfo[] sensors1 = []; + SensorInfo[] sensors2 = []; + + var url1 = $"http://{ip}/get_sensors_info?page=1"; + var url2 = $"http://{ip}/get_sensors_info?page=2"; + + + var task1 = cumulus.MyHttpClient.GetStringAsync(url1, token); + var task2 = cumulus.MyHttpClient.GetStringAsync(url2, token); + + // Wait for both tasks to complete + await Task.WhenAll(task1, task2); + + // Retrieve the results + string result1 = await task1; + string result2 = await task2; + + cumulus.LogDataMessage("GetSensorInfo: Page 1 = " + result1); + cumulus.LogDataMessage("GetSensorInfo: Page 2 = " + result2); + + if (!string.IsNullOrEmpty(result1)) + { + sensors1 = result1.FromJson(); + } + if (!string.IsNullOrEmpty(result2)) + { + sensors2 = result2.FromJson(); + } + + var retArr = new SensorInfo[sensors1.Length + sensors2.Length]; + sensors1.CopyTo(retArr, 0); + sensors2.CopyTo(retArr, sensors1.Length); + + return retArr; } @@ -208,36 +419,7 @@ public void Dispose() } } - private enum commonSensorTypes - { - indoorTemp = 1, - temp = 2, - dewpoint = 3, - windchill = 4, - heatindex = 5, - indoorHum = 6, - hum = 7, - absPressure = 8, - relPressure = 9, - windDir = 10, - windSpeed = 11, - windGust = 12, - rainEvent = 13, - rainRate = 14, - rainGain = 15, - rainDay = 16, - rainWeek = 17, - rainMonth = 18, - rainYear = 19, - rainTotals = 20, - light = 21, - uv = 22, - uvindex = 23, - time = 24, - dayWindMax = 25 - } - - public class commonSensor + public class CommonSensor { public string id { get; set; } public string val { get; set; } @@ -269,7 +451,7 @@ public double? valDbl } } - public class tempHumSensor + public class TempHumSensor { public int channel { get; set; } public int? battery { get; set; } @@ -287,7 +469,7 @@ public int? humidityVal } - public class wh25Sensor + public class Wh25Sensor { public double intemp { get; set; } public string unit { get; set; } @@ -305,7 +487,7 @@ public int? inhumiInt } } - public class lightningSensor + public class LightningSensor { public string distance { get; set; } public string timestamp { get; set; } @@ -331,7 +513,7 @@ public string distanceUnit } } - public class co2Sensor + public class Co2Sensor { public double? temp { get; set; } public string unit { get; set; } @@ -355,7 +537,7 @@ public int? humidityVal } } - public class ch_pm25Sensor + public class Ch_Pm25Sensor { public int? channel { get; set; } public double? PM25 { get; set; } @@ -364,7 +546,7 @@ public class ch_pm25Sensor public int? battery { get; set; } } - public class ch_leakSensor + public class Ch_LeakSensor { public int? channel { get; set; } public string name { get; set; } @@ -372,20 +554,31 @@ public class ch_leakSensor public string status { get; set; } } - public class liveData + public class LiveData { - public commonSensor[] common_list { get; set; } - public commonSensor[]? rain { get; set; } - public commonSensor[]? piezoRain { get; set; } - public wh25Sensor[]? wh25 { get; set; } - public lightningSensor[]? lightning { get; set; } - public co2Sensor[]? co2 { get; set; } - public ch_pm25Sensor[]? ch_pm25 { get; set; } - public ch_leakSensor[]? ch_leak { get; set; } - public tempHumSensor[]? ch_aisle { get; set; } - public tempHumSensor[]? ch_soil { get; set; } - public tempHumSensor[]? ch_temp { get; set; } - public tempHumSensor[]? ch_leaf { get; set; } + public CommonSensor[] common_list { get; set; } + public CommonSensor[]? rain { get; set; } + public CommonSensor[]? piezoRain { get; set; } + public Wh25Sensor[]? wh25 { get; set; } + public LightningSensor[]? lightning { get; set; } + public Co2Sensor[]? co2 { get; set; } + public Ch_Pm25Sensor[]? ch_pm25 { get; set; } + public Ch_LeakSensor[]? ch_leak { get; set; } + public TempHumSensor[]? ch_aisle { get; set; } + public TempHumSensor[]? ch_soil { get; set; } + public TempHumSensor[]? ch_temp { get; set; } + public TempHumSensor[]? ch_leaf { get; set; } + } + + public class SensorInfo + { + public string img { get; set; } + public int type { get; set; } + public string name { get; set; } + public string id { get; set; } + public int batt { get; set; } + public int signal { get; set; } + public bool idst { get; set; } } } } diff --git a/CumulusMX/WeatherStation.cs b/CumulusMX/WeatherStation.cs index 0f89fcc4..14a37d44 100644 --- a/CumulusMX/WeatherStation.cs +++ b/CumulusMX/WeatherStation.cs @@ -197,6 +197,8 @@ public struct DayFileRec public WeatherDataCollection weatherDataCollection = []; + public List LowBatteryDevices = new List(); + // Current values public double THWIndex = 0; diff --git a/CumulusMX/webtags.cs b/CumulusMX/webtags.cs index 5dfea56d..1233a8cd 100644 --- a/CumulusMX/webtags.cs +++ b/CumulusMX/webtags.cs @@ -1091,6 +1091,11 @@ private string Tagtxbattery(Dictionary tagParams) return station.TxBatText; } + private string TagLowBatteryList(Dictionary tagParams) + { + return string.Join(",", station.LowBatteryDevices); + } + private string TagMulticastGoodCnt(Dictionary tagParams) { return station.multicastsGood.ToString(); @@ -5974,6 +5979,7 @@ public void InitialiseWebtags() { "battery", Tagbattery }, { "txbattery", Tagtxbattery }, { "ConsoleSupplyV", TagConsoleSupplyV }, + { "LowBatteryList", TagLowBatteryList }, { "MulticastBadCnt", TagMulticastBadCnt }, { "MulticastGoodCnt", TagMulticastGoodCnt }, { "MulticastGoodPct", TagMulticastGoodPct }, diff --git a/Updates.txt b/Updates.txt index c495887e..5fa86f41 100644 --- a/Updates.txt +++ b/Updates.txt @@ -1,9 +1,17 @@ 4.2.0 - b4031 ————————————— New +- New station type: Ecowitt Local HTTP API + - Ecowitt Local HTTP API is an alternative the local TCP API used by the gateways and some stations + - Currently it is less capable than the TCP API + - Allows direct support of Ecowitt stations that do not support the TCP API and currently have to use the Custom HTTP Server mode - Exposes the UseDataLogger setting in Station Settings > Options > Advanced - Implements a new data viewer where you can select and view historic data from the monthly logs for a given period, for a set of data values. See: "Data logs > Interval Data Viewer" - Implements a new data viewer where you can select and view historic daily data from the day file for a given period, for a set of data values. See: "Data logs > Daily Data Viewer" +- New web tag <#LowBatteryList> which returns the list of sensors/transmitters that have low batteries + - The format is a comma separated list of sensor/transmitter IDs and battery states + - Eg. "wh80 - ,wh41ch1-,wh41ch2-" + - Where is "LOW" or "0" or "1" depending on what the sensor sends Changed - The Station Settings screen now does a two stage selection: First the manufacturer, then the station model. This shortens the long and list to select from. From d2885cbbef036436f931fe9b7d4c88f72bdaf08d Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Mon, 2 Sep 2024 23:06:59 +0100 Subject: [PATCH 20/63] Code tidy --- CumulusMX/EcowittHttpApiStation.cs | 25 +++++-------------------- CumulusMX/EcowittLocalApi.cs | 8 ++++---- 2 files changed, 9 insertions(+), 24 deletions(-) diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index 5806b0d3..688f2d0c 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -1,13 +1,9 @@ using System; -using System.Collections.Generic; using System.Linq; using System.Threading; using System.Threading.Tasks; using System.Timers; -using static System.Runtime.InteropServices.JavaScript.JSType; -using static CumulusMX.GW1000Api; - namespace CumulusMX { @@ -15,8 +11,6 @@ namespace CumulusMX internal class EcowittHttpApiStation : WeatherStation #pragma warning restore CA1001 // Types that own disposable fields should be disposable { - private readonly string ipaddr; - private readonly string macaddr; private string deviceModel; private string deviceFirmware; private readonly int updateRate = 10000; // 10 seconds by default @@ -111,9 +105,6 @@ public EcowittHttpApiStation(Cumulus cumulus) : base(cumulus) cumulus.LogMessage("Using the piezo rain sensor data"); } - ipaddr = cumulus.Gw1000IpAddress; - macaddr = cumulus.Gw1000MacAddress; - localApi = new EcowittLocalApi(cumulus); ecowittApi = new EcowittApi(cumulus, this); @@ -152,7 +143,7 @@ public override void Start() { try { - var dataLastRead = DateTime.MinValue; + DateTime dataLastRead; double delay; while (!cumulus.cancellationToken.IsCancellationRequested) @@ -183,7 +174,7 @@ public override void Start() if (rawData.co2 != null) { - ProcessCo2(rawData.co2, dataLastRead); + ProcessCo2(rawData.co2); } if (rawData.ch_pm25 != null) @@ -502,7 +493,6 @@ private async Task GetSensorIds() var sensors = await localApi.GetSensorInfo(cumulus.cancellationToken); batteryLow = false; - bool sensorOk; LowBatteryDevices.Clear(); @@ -1003,7 +993,7 @@ private void ProcessLightning(EcowittLocalApi.LightningSensor[] sensors, DateTim } } - private void ProcessCo2(EcowittLocalApi.Co2Sensor[] sensors, DateTime dateTime) + private void ProcessCo2(EcowittLocalApi.Co2Sensor[] sensors) { cumulus.LogDebugMessage("WH45 CO₂: Decoding..."); @@ -1050,7 +1040,7 @@ private void ProcessCo2(EcowittLocalApi.Co2Sensor[] sensors, DateTime dateTime) CO2_pm10_aqi = GetAqi(AqMeasure.pm10, CO2_pm10); } - private void ProcessChPm25(EcowittLocalApi.Ch_Pm25Sensor[] sensors) + private void ProcessChPm25(EcowittLocalApi.ChPm25Sensor[] sensors) { cumulus.LogDebugMessage($"ProcessChPm25: Processing {sensors.Length} sensors"); @@ -1072,7 +1062,7 @@ private void ProcessChPm25(EcowittLocalApi.Ch_Pm25Sensor[] sensors) } } - private void ProcessLeak(EcowittLocalApi.Ch_LeakSensor[] sensors) + private void ProcessLeak(EcowittLocalApi.ChLeakSensor[] sensors) { cumulus.LogDebugMessage($"ProcessLeak: Processing {sensors.Length} sensors"); @@ -1208,11 +1198,6 @@ private void GetSystemInfo(bool driftOnly) } - private static string TestBattery1(int value) - { - return value == 0 ? "OK" : "Low"; - } - private static bool TestBattery3(int value) { if (value == 6) diff --git a/CumulusMX/EcowittLocalApi.cs b/CumulusMX/EcowittLocalApi.cs index cc77e2d1..e5f3f5e3 100644 --- a/CumulusMX/EcowittLocalApi.cs +++ b/CumulusMX/EcowittLocalApi.cs @@ -537,7 +537,7 @@ public int? humidityVal } } - public class Ch_Pm25Sensor + public class ChPm25Sensor { public int? channel { get; set; } public double? PM25 { get; set; } @@ -546,7 +546,7 @@ public class Ch_Pm25Sensor public int? battery { get; set; } } - public class Ch_LeakSensor + public class ChLeakSensor { public int? channel { get; set; } public string name { get; set; } @@ -562,8 +562,8 @@ public class LiveData public Wh25Sensor[]? wh25 { get; set; } public LightningSensor[]? lightning { get; set; } public Co2Sensor[]? co2 { get; set; } - public Ch_Pm25Sensor[]? ch_pm25 { get; set; } - public Ch_LeakSensor[]? ch_leak { get; set; } + public ChPm25Sensor[]? ch_pm25 { get; set; } + public ChLeakSensor[]? ch_leak { get; set; } public TempHumSensor[]? ch_aisle { get; set; } public TempHumSensor[]? ch_soil { get; set; } public TempHumSensor[]? ch_temp { get; set; } From 0db6196f49ddcc3fd61b1d40e12d951b9bf38ab9 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Thu, 5 Sep 2024 10:02:27 +0100 Subject: [PATCH 21/63] Added VP2 battery status to new web tag #LowBatteryList Fixed not logging PHP upload failures to the warning log --- CumulusMX/Cumulus.cs | 15 +++++++++------ CumulusMX/DavisStation.cs | 15 ++++++++++++++- CumulusMX/WeatherStation.cs | 2 +- Updates.txt | 2 +- 4 files changed, 25 insertions(+), 9 deletions(-) diff --git a/CumulusMX/Cumulus.cs b/CumulusMX/Cumulus.cs index 0c70c69d..4b246d5f 100644 --- a/CumulusMX/Cumulus.cs +++ b/CumulusMX/Cumulus.cs @@ -11077,9 +11077,12 @@ private async Task UploadString(HttpClient httpclient, bool incremental, s LogDebugMessage($"{prefix}: Uploading to {remotefile}"); // we will try this twice in case the first attempt fails - var retry = 0; + var retry = -1; do { + // retry == 0 is the initial upload + retry++; + MemoryStream outStream = null; StreamContent streamContent = null; @@ -11202,17 +11205,18 @@ private async Task UploadString(HttpClient httpclient, bool incremental, s { LogMessage($"{prefix}: Upload to {remotefile}: Response text follows:\n{responseBodyAsText}"); - retry++; - if (retry >= 2) + if (retry == 2) { LogWarningMessage($"{prefix}: HTTP Error uploading to {remotefile}: Response code = {(int) response.StatusCode}: {response.StatusCode}"); - return false; + } + else + { + await Task.Delay(2000); } } } catch (HttpRequestException ex) { - retry++; if (retry < 2) { LogDebugMessage($"{prefix}: HTTP Error uploading to {remotefile} - " + ex.Message); @@ -11228,7 +11232,6 @@ private async Task UploadString(HttpClient httpclient, bool incremental, s } catch (Exception ex) { - retry++; if (retry < 2) { if (ex.InnerException is TimeoutException) diff --git a/CumulusMX/DavisStation.cs b/CumulusMX/DavisStation.cs index fe599250..aa7f7dcf 100644 --- a/CumulusMX/DavisStation.cs +++ b/CumulusMX/DavisStation.cs @@ -8,6 +8,8 @@ using System.Text; using System.Threading; +using static CumulusMX.Tempest.RestPacket; + namespace CumulusMX { internal class DavisStation : WeatherStation @@ -1830,8 +1832,15 @@ private void GetAndProcessLoopData(int number) DoForecast(forecast, false); + LowBatteryDevices.Clear(); + ConBatText = loopData.ConBatVoltage.ToString("F2"); + if (loopData.ConBatVoltage <= 3.5) + { + LowBatteryDevices.Add("Console-" + ConBatText); + } + TxBatText = ProcessTxBatt(loopData.TXbattStatus); cumulus.BatteryLowAlarm.Triggered = TxBatText.Contains("LOW") || loopData.ConBatVoltage <= 3.5; @@ -1990,13 +1999,17 @@ private void GetAndProcessLoopData(int number) } } - private static string ProcessTxBatt(byte txStatus) + private string ProcessTxBatt(byte txStatus) { StringBuilder response = new(); for (int i = 0; i < 8; i++) { var status = (txStatus & (1 << i)) == 0 ? "-ok " : "-LOW "; + if (status == "-LOW") + { + LowBatteryDevices.Add((i + 1) + status); + } response.Append((i + 1) + status); } diff --git a/CumulusMX/WeatherStation.cs b/CumulusMX/WeatherStation.cs index 14a37d44..2bb39d47 100644 --- a/CumulusMX/WeatherStation.cs +++ b/CumulusMX/WeatherStation.cs @@ -197,7 +197,7 @@ public struct DayFileRec public WeatherDataCollection weatherDataCollection = []; - public List LowBatteryDevices = new List(); + public List LowBatteryDevices { get; set; } = []; // Current values diff --git a/Updates.txt b/Updates.txt index 5fa86f41..083b1f4c 100644 --- a/Updates.txt +++ b/Updates.txt @@ -20,7 +20,7 @@ Changed Fixed - Davis Cloud stations no longer continuosly try to fetch history data if there is no Pro subscription - WMR928 Station now correctly converts indoor temperatures to the user defined units - +- No logging PHP upload failures to the warning log From 65ac9a63dcebca13cfa54a34c893a1b62e206fd6 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Thu, 5 Sep 2024 10:09:25 +0100 Subject: [PATCH 22/63] Add WLL #LowBatteryList handling --- CumulusMX/DavisWllStation.cs | 15 +++++++++++++++ 1 file changed, 15 insertions(+) diff --git a/CumulusMX/DavisWllStation.cs b/CumulusMX/DavisWllStation.cs index 849d93d8..0be4982b 100644 --- a/CumulusMX/DavisWllStation.cs +++ b/CumulusMX/DavisWllStation.cs @@ -2990,6 +2990,21 @@ private void GetWlHistoricHealth() cumulus.LogErrorMessage("WLL Health: exception: " + ex.Message); } cumulus.BatteryLowAlarm.Triggered = TxBatText.Contains("LOW") || wllVoltageLow; + + // Just the low battery list + LowBatteryDevices.Clear(); + if (wllVoltageLow) + { + LowBatteryDevices.Add("Console-" + ConBatText); + } + var arr = TxBatText.Split(' '); + for (int i = 0; i < arr.Length; i++) + { + if (arr[i].Contains("LOW")) + { + LowBatteryDevices.Add(arr[i]); + } + } } catch (Exception ex) { From 3c2237d19c9f0b1d7b27bc34f86034fd443e263a Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Thu, 5 Sep 2024 10:12:03 +0100 Subject: [PATCH 23/63] Add WLL #LowBatteryList handling --- CumulusMX/DavisWllStation.cs | 15 +++++++++++++-- 1 file changed, 13 insertions(+), 2 deletions(-) diff --git a/CumulusMX/DavisWllStation.cs b/CumulusMX/DavisWllStation.cs index 0be4982b..d137cf09 100644 --- a/CumulusMX/DavisWllStation.cs +++ b/CumulusMX/DavisWllStation.cs @@ -1313,9 +1313,20 @@ private void DecodeCurrent(string currentJson) // If the station isn't using the logger function for WLL - i.e. no API key, then only alarm on Tx battery status // otherwise, trigger the alarm when we read the Health data which also contains the WLL backup battery status - if (!cumulus.StationOptions.UseDataLogger) + LowBatteryDevices.Clear(); + + if (!cumulus.StationOptions.UseDataLogger && TxBatText.Contains("LOW")) { - cumulus.BatteryLowAlarm.Triggered = TxBatText.Contains("LOW"); + cumulus.BatteryLowAlarm.Triggered = true; + // Just the low battery list + var arr = TxBatText.Split(' '); + for (int i = 0; i < arr.Length; i++) + { + if (arr[i].Contains("LOW")) + { + LowBatteryDevices.Add(arr[i]); + } + } } } catch (Exception exp) From 0c1aa9bef1223d54fefd86a591d62452ef262a9b Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Thu, 5 Sep 2024 10:19:07 +0100 Subject: [PATCH 24/63] WL Cloud - some incomplete support for #LowBatteryList --- CumulusMX/DavisCloudStation.cs | 15 +++++++++++++-- 1 file changed, 13 insertions(+), 2 deletions(-) diff --git a/CumulusMX/DavisCloudStation.cs b/CumulusMX/DavisCloudStation.cs index a9f03485..58d2936e 100644 --- a/CumulusMX/DavisCloudStation.cs +++ b/CumulusMX/DavisCloudStation.cs @@ -2110,9 +2110,20 @@ private void DecodeCurrent(List sensors) // If the station isn't using the logger function for WLL - i.e. no API key, then only alarm on Tx battery status // otherwise, trigger the alarm when we read the Health data which also contains the WLL backup battery status - if (!cumulus.StationOptions.UseDataLogger) + LowBatteryDevices.Clear(); + + if (!cumulus.StationOptions.UseDataLogger && TxBatText.Contains("LOW")) { - cumulus.BatteryLowAlarm.Triggered = TxBatText.Contains("LOW"); + cumulus.BatteryLowAlarm.Triggered = true; + // Just the low battery list + var arr = TxBatText.Split(' '); + for (int i = 0; i < arr.Length; i++) + { + if (arr[i].Contains("LOW")) + { + LowBatteryDevices.Add(arr[i]); + } + } } } From cff2eed769d956e9a63ab5ec1d5e5541633eeeba Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Thu, 5 Sep 2024 10:19:34 +0100 Subject: [PATCH 25/63] Updates.txt --- Updates.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Updates.txt b/Updates.txt index 083b1f4c..695ec0b8 100644 --- a/Updates.txt +++ b/Updates.txt @@ -20,7 +20,7 @@ Changed Fixed - Davis Cloud stations no longer continuosly try to fetch history data if there is no Pro subscription - WMR928 Station now correctly converts indoor temperatures to the user defined units -- No logging PHP upload failures to the warning log +- Not logging PHP upload failures to the warning log From 6ba95b13c4afa210ef4ff72dd7e79a16c3e269be Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sun, 8 Sep 2024 12:05:34 +0100 Subject: [PATCH 26/63] Add Ecowitt HTTP API password Add polling time update to Ecowitt HTTP API station Add more Ecowitt HTTP API station error handling Ecowitt HTTP API station ignore apparent temp --- CumulusMX/Cumulus.cs | 7 + CumulusMX/CumulusMX.csproj | 3 +- CumulusMX/EcowittHttpApiStation.cs | 175 +++++++++++++++- CumulusMX/EcowittLocalApi.cs | 321 ++++++++++++++++++++++++----- CumulusMX/StationSettings.cs | 5 +- CumulusMX/Utils.cs | 2 +- Updates.txt | 32 +-- 7 files changed, 470 insertions(+), 75 deletions(-) diff --git a/CumulusMX/Cumulus.cs b/CumulusMX/Cumulus.cs index 4b246d5f..53db9bf1 100644 --- a/CumulusMX/Cumulus.cs +++ b/CumulusMX/Cumulus.cs @@ -3975,6 +3975,8 @@ private void ReadIniFile() EcowittLocalAddr = ini.GetValue("GW1000", "EcowittLocalAddr", localIp.ToString()); EcowittCustomInterval = ini.GetValue("GW1000", "EcowittCustomInterval", 16, 1); + EcowittHttpPassword = ini.GetValue("GW1000", "HttpPassword", ""); + EcowittExtraSetCustomServer = ini.GetValue("GW1000", "ExtraSetCustomServer", false); EcowittExtraGatewayAddr = ini.GetValue("GW1000", "EcowittExtraGwAddr", "0.0.0.0"); EcowittExtraLocalAddr = ini.GetValue("GW1000", "EcowittExtraLocalAddr", localIp.ToString()); @@ -5380,6 +5382,7 @@ .. etc HTTPProxyPassword = Crypto.DecryptString(HTTPProxyPassword, Program.InstanceId, "HTTPProxyPassword"); EcowittApplicationKey = Crypto.DecryptString(EcowittApplicationKey, Program.InstanceId, "EcowittSettings.AppKey"); EcowittUserApiKey = Crypto.DecryptString(EcowittUserApiKey, Program.InstanceId, "EcowittSettings.UserApiKey"); + EcowittHttpPassword = Crypto.DecryptString(EcowittHttpPassword, Program.InstanceId, "EcowittSettings.HttpPassword"); } else { @@ -5672,6 +5675,9 @@ internal void WriteIniFile() ini.SetValue("GW1000", "EcowittExtraGwAddr", EcowittExtraGatewayAddr); ini.SetValue("GW1000", "EcowittExtraLocalAddr", EcowittExtraLocalAddr); ini.SetValue("GW1000", "EcowittExtraCustomInterval", EcowittExtraCustomInterval); + + ini.SetValue("GW1000", "HttpPassword", Crypto.EncryptString(EcowittHttpPassword, Program.InstanceId, "EcowittSettings.HttpPassword")); + // api ini.SetValue("GW1000", "EcowittAppKey", Crypto.EncryptString(EcowittApplicationKey, Program.InstanceId, "EcowittSettings.AppKey")); ini.SetValue("GW1000", "EcowittUserKey", Crypto.EncryptString(EcowittUserApiKey, Program.InstanceId, "EcowittSettings.UserApiKey")); @@ -7159,6 +7165,7 @@ public void WriteStringsFile() public string[] EcowittForwarders { get; set; } = new string[10]; public bool EcowittExtraUseMainForwarders { get; set; } public string[] EcowittExtraForwarders { get; set; } = new string[10]; + public string EcowittHttpPassword { get; set; } public int[] EcowittMapWN34 { get; set; } = new int[9]; diff --git a/CumulusMX/CumulusMX.csproj b/CumulusMX/CumulusMX.csproj index 2ad57592..8ae6b6e7 100644 --- a/CumulusMX/CumulusMX.csproj +++ b/CumulusMX/CumulusMX.csproj @@ -55,8 +55,9 @@ - 4.2.0.4031 + 4.2.0.4033-beta Copyright © 2015-$([System.DateTime]::Now.ToString('yyyy')) Cumulus MX + $(AssemblyName) diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index 688f2d0c..5a7724fc 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -13,7 +13,7 @@ internal class EcowittHttpApiStation : WeatherStation { private string deviceModel; private string deviceFirmware; - private readonly int updateRate = 10000; // 10 seconds by default + private int updateRate = 10000; // 10 seconds by default private int lastMinute = -1; private int lastHour = -1; @@ -141,6 +141,8 @@ public override void Start() liveTask = Task.Run(() => { + var excepMsg = "unknown error"; + try { DateTime dataLastRead; @@ -325,13 +327,25 @@ public override void Start() } } // Catch the ThreadAbortException - catch (ThreadAbortException) + catch (ThreadAbortException ex) { //do nothing + excepMsg = ex.Message; + } + catch (Exception ex) + { + excepMsg = ex.Message; } finally { - cumulus.LogMessage("Local API task ended"); + if (cumulus.cancellationToken.IsCancellationRequested) + { + cumulus.LogMessage("Local API task cancelled by request"); + } + else + { + cumulus.LogCriticalMessage("Local API task ended unexpectedly: " + excepMsg); + } } }, cumulus.cancellationToken); } @@ -518,6 +532,12 @@ private async Task GetSensorIds() goto case 1003; case 2: // wh80 name = "wh80"; + // if a WS80 is connected, it has a 4.75 second update rate, so reduce the MX update rate from the default 10 seconds + if (updateRate > 4000 && updateRate != 4000) + { + cumulus.LogMessage($"GetSensorIds: WS80 sensor detected, changing the update rate from {(updateRate / 1000):D} seconds to 4 seconds"); + updateRate = 4000; + } goto case 1003; case 3: // wh40 name = "wh40"; @@ -553,9 +573,21 @@ private async Task GetSensorIds() goto case 1003; case 48: // wh90 name = "wh90"; + // if a WS90 is connected, it has a 8.8 second update rate, so reduce the MX update rate from the default 10 seconds + if (updateRate > 8000 && updateRate != 8000) + { + cumulus.LogMessage($"GetSensorIds: WS90 sensor detected, changing the update rate from {(updateRate / 1000):D} seconds to 8 seconds"); + updateRate = 8000; + } goto case 1003; case 49: // wh85 name = "wh85"; + // if a WH85 is connected, it has a 8.5 second update rate, so reduce the MX update rate from the default 10 seconds + if (updateRate > 8000 && updateRate != 8000) + { + cumulus.LogMessage($"GetSensorIds: WH85 sensor detected, changing the update rate from {(updateRate / 1000):D} seconds to 8 seconds"); + updateRate = 8000; + } goto case 1003; @@ -628,6 +660,9 @@ private void ProcessCommonList(EcowittLocalApi.CommonSensor[] sensors, DateTime windchill = sensor.unit == "C" ? ConvertUnits.TempCToUser(windchill) : ConvertUnits.TempFToUser(windchill); } break; + case "4": //Apparent + // do nothing with this for now - MX calcuates apparent + break; case "0x05": //Heat index // cumulus calculates this break; @@ -729,6 +764,14 @@ private void ProcessCommonList(EcowittLocalApi.CommonSensor[] sensors, DateTime private void ProcessWh25(EcowittLocalApi.Wh25Sensor[] sensors, DateTime dateTime) { + //"wh25": [{ + // "intemp": "23.8", + // "unit": "C", + // "inhumi": "68%", + // "abs": "1006.5 hPa", + // "rel": "1010.5 hPa" + //}] + for (var i = 0; i < sensors.Length; i++) { var sensor = sensors[i]; @@ -821,6 +864,35 @@ private void ProcessWh25(EcowittLocalApi.Wh25Sensor[] sensors, DateTime dateTime private void ProcessRain(EcowittLocalApi.CommonSensor[] sensors) { + //"rain"/"piezoRain": [ + // { + // "id": "0x0D", + // "val": "0.0 mm" + // }, + // { + // "id": "0x0E", + // "val": "0.0 mm/Hr" + // }, + // { + // "id": "0x10", + // "val": "0.0 mm" + // }, + // { + // "id": "0x11", + // "val": "5.0 mm" + // }, + // { + // "id": "0x12", + // "val": "27.1 mm" + // }, + // { + // "id": "0x13", + // "val": "681.4 mm", + // "battery": "5" + // } + //], + + for (var i = 0; i < sensors.Length; i++) { var sensor = sensors[i]; @@ -928,6 +1000,12 @@ private void ProcessRain(EcowittLocalApi.CommonSensor[] sensors) private void ProcessLightning(EcowittLocalApi.LightningSensor[] sensors, DateTime dateTime) { + // "lightning": [{ + // "distance": "16.7 mi", + // "timestamp": "09/01/2024 18:45:14", + // "count": "0", + // "battery": "5" + // }] for (var i = 0; i < sensors.Length; i++) { try @@ -951,10 +1029,11 @@ private void ProcessLightning(EcowittLocalApi.LightningSensor[] sensors, DateTim //Lightning time (UTC) if (!string.IsNullOrEmpty(sensor.timestamp)) { - // oh my god, it sends the time as "MM/dd/yyyy HH: mm: ss" - // TODO: what is default time if not strikes detecetd yet? - var arr = sensor.timestamp.Split(' '); + // oh my god, it sends the time as "MM/dd/yyyy HH: mm: ss" for some locales + // TODO: what is default time if not strikes detected yet? + var arr = sensor.timestamp.Replace(": ", ":").Split(' '); var date = arr[0].Split('/'); + var time = arr[1].Split(':'); //if (sensor.timestamp == "default string") //{ @@ -966,9 +1045,9 @@ private void ProcessLightning(EcowittLocalApi.LightningSensor[] sensors, DateTim int.Parse(date[2]), int.Parse(date[0]), int.Parse(date[1]), - int.Parse(arr[1][0..^1]), - int.Parse(arr[2][0..^1]), - int.Parse(arr[3][0..^1]), + int.Parse(time[0]), + int.Parse(time[1]), + int.Parse(time[2]), 0, DateTimeKind.Utc); //} } @@ -995,6 +1074,23 @@ private void ProcessLightning(EcowittLocalApi.LightningSensor[] sensors, DateTim private void ProcessCo2(EcowittLocalApi.Co2Sensor[] sensors) { + // "co2": [ + // { + // "temp": "24.4", + // "unit": "C", + // "humidity": "62%", + // "PM25": "0.9", + // "PM25_RealAQI": "4", + // "PM25_24HAQI": "7", + // "PM10": "0.9", + // "PM10_RealAQI": "1", + // "PM10_24HAQI": "2", + // "CO2": "323", + // "CO2_24H": "348", + // "battery": "6" + // } + // ] + cumulus.LogDebugMessage("WH45 CO₂: Decoding..."); for (var i = 0; i < sensors.Length; i++) @@ -1042,6 +1138,16 @@ private void ProcessCo2(EcowittLocalApi.Co2Sensor[] sensors) private void ProcessChPm25(EcowittLocalApi.ChPm25Sensor[] sensors) { + //"ch_pm25": [ + // { + // "channel": "1", + // "PM25": "6.0", + // "PM25_RealAQI": "25", + // "PM25_24HAQI": "24", + // "battery": "5" + // } + //] + cumulus.LogDebugMessage($"ProcessChPm25: Processing {sensors.Length} sensors"); for (var i = 0; i < sensors.Length; i++) @@ -1064,6 +1170,15 @@ private void ProcessChPm25(EcowittLocalApi.ChPm25Sensor[] sensors) private void ProcessLeak(EcowittLocalApi.ChLeakSensor[] sensors) { + //"ch_leak": [ + // { + // "channel": "2", + // "name": "", + // "battery": "4", + // "status": "Normal" + // } + //] + cumulus.LogDebugMessage($"ProcessLeak: Processing {sensors.Length} sensors"); for (var i = 0; i < sensors.Length; i++) @@ -1088,6 +1203,17 @@ private void ProcessLeak(EcowittLocalApi.ChLeakSensor[] sensors) private void ProcessExtraTempHum(EcowittLocalApi.TempHumSensor[] sensors, DateTime dateTime) { + //"ch_aisle": [ + // { + // "channel": "1", + // "name": "", + // "battery": "0", + // "temp": "24.9", + // "unit": "C", + // "humidity": "61%" + // } + //] + cumulus.LogDebugMessage($"ProcessExtraTempHum: Processing {sensors.Length} sensors"); for (var i = 0; i < sensors.Length; i++) @@ -1119,6 +1245,16 @@ private void ProcessExtraTempHum(EcowittLocalApi.TempHumSensor[] sensors, DateTi private void ProcessUserTemp(EcowittLocalApi.TempHumSensor[] sensors) { + //"ch_temp": [ + // { + // "channel": "1", + // "name": "", + // "temp": "21.5", + // "unit": "C", + // "battery": "3" + // } + //] + // user temp = WH34 8 channel Soil or Water temperature sensors cumulus.LogDebugMessage($"ProcessUserTemp: Processing {sensors.Length} sensors"); @@ -1150,6 +1286,15 @@ private void ProcessUserTemp(EcowittLocalApi.TempHumSensor[] sensors) private void ProcessSoilMoisture(EcowittLocalApi.TempHumSensor[] sensors) { + //"ch_soil": [ + // { + // "channel": "1", + // "name": "", + // "battery": "5", + // "humidity": "56%" + // } + //] + cumulus.LogDebugMessage($"ProcessSoilMoisture: Processing {sensors.Length} sensors"); for (var i = 0; i < sensors.Length; i++) @@ -1172,6 +1317,15 @@ private void ProcessSoilMoisture(EcowittLocalApi.TempHumSensor[] sensors) private void ProcessLeafWet(EcowittLocalApi.TempHumSensor[] sensors) { + //"ch_leaf": [ + // { + // "channel": "1", + // "name": "CH1 Leaf Wetness", + // "humidity": "10%", + // "battery": "5" + // } + //] + cumulus.LogDebugMessage($"ProcessLeafWet: Processing {sensors.Length} sensors"); for (var i = 0; i < sensors.Length; i++) @@ -1195,7 +1349,6 @@ private void ProcessLeafWet(EcowittLocalApi.TempHumSensor[] sensors) private void GetSystemInfo(bool driftOnly) { cumulus.LogMessage("NOT Reading Ecowitt system info"); - } private static bool TestBattery3(int value) @@ -1228,7 +1381,7 @@ private void DataTimeout(object source, ElapsedEventArgs e) } else { - cumulus.LogErrorMessage($"ERROR: No data received from the sttaion for {tmrDataWatchdog.Interval / 1000} seconds"); + cumulus.LogErrorMessage($"ERROR: No data received from the station for {tmrDataWatchdog.Interval / 1000} seconds"); if (!DataStopped) { DataStoppedTime = DateTime.Now; diff --git a/CumulusMX/EcowittLocalApi.cs b/CumulusMX/EcowittLocalApi.cs index e5f3f5e3..42c6f372 100644 --- a/CumulusMX/EcowittLocalApi.cs +++ b/CumulusMX/EcowittLocalApi.cs @@ -5,7 +5,6 @@ using ServiceStack; - namespace CumulusMX { internal sealed class EcowittLocalApi : IDisposable @@ -148,41 +147,54 @@ public LiveData GetLiveData(CancellationToken token) int responseCode; int retries = 3; - string ip = cumulus.Gw1000IpAddress; int retry = 1; - do + if (!Utils.ValidateIPv4(cumulus.Gw1000IpAddress)) { - var url = $"http://{ip}/get_livedata_info"; + cumulus.LogErrorMessage("GetLiveData: Invalid station IP address: " + cumulus.Gw1000IpAddress); + return null; + } - // we want to do this synchronously, so .Result - using (var response = cumulus.MyHttpClient.GetAsync(url, token).Result) - { - responseBody = response.Content.ReadAsStringAsync(token).Result; - responseCode = (int) response.StatusCode; - cumulus.LogDebugMessage($"LocalApi.GetLiveData: Ecowitt Local API GetLiveData Response code: {responseCode}"); - cumulus.LogDataMessage($"LocalApi.GetLiveData: Ecowitt Local API GetLiveData Response: {responseBody}"); - } - if (responseCode != 200) + do + { + try { - cumulus.LogWarningMessage($"LocalApi.GetLiveData: Ecowitt Local API GetLiveData Error: {responseCode}"); - Cumulus.LogConsoleMessage($" - Error {responseCode}", ConsoleColor.Red); - return null; - } + var url = $"http://{cumulus.Gw1000IpAddress}/get_livedata_info"; + // we want to do this synchronously, so .Result + using (var response = cumulus.MyHttpClient.GetAsync(url, token).Result) + { + responseBody = response.Content.ReadAsStringAsync(token).Result; + responseCode = (int) response.StatusCode; + cumulus.LogDebugMessage($"LocalApi.GetLiveData: Ecowitt Local API GetLiveData Response code: {responseCode}"); + cumulus.LogDataMessage($"LocalApi.GetLiveData: Ecowitt Local API GetLiveData Response: {responseBody}"); + } - if (responseBody == "{}") - { - cumulus.LogMessage("LocalApi.GetLiveData: Ecowitt Local API GetLiveData: No data was returned."); - Cumulus.LogConsoleMessage(" - No Live data available"); - return null; + if (responseCode != 200) + { + cumulus.LogWarningMessage($"LocalApi.GetLiveData: Ecowitt Local API GetLiveData Error: {responseCode}"); + Cumulus.LogConsoleMessage($" - Error {responseCode}", ConsoleColor.Red); + return null; + } + + + if (responseBody == "{}") + { + cumulus.LogMessage("LocalApi.GetLiveData: Ecowitt Local API GetLiveData: No data was returned."); + Cumulus.LogConsoleMessage(" - No Live data available"); + return null; + } + else if (responseBody.StartsWith('{')) // sanity check + { + // Convert JSON string to an object + LiveData json = responseBody.FromJson(); + return json; + } } - else if (responseBody.StartsWith('{')) // sanity check + catch (Exception ex) { - // Convert JSON string to an object - LiveData json = responseBody.FromJson(); - return json; + cumulus.LogExceptionMessage(ex, "GetLiveData: Error"); } } while (retries-- > 0); @@ -369,42 +381,257 @@ .. chans 2-8 */ - string ip = cumulus.Gw1000IpAddress; + if (!Utils.ValidateIPv4(cumulus.Gw1000IpAddress)) + { + cumulus.LogErrorMessage("GetSensorInfo: Invalid station IP address: " + cumulus.Gw1000IpAddress); + return null; + } SensorInfo[] sensors1 = []; SensorInfo[] sensors2 = []; - var url1 = $"http://{ip}/get_sensors_info?page=1"; - var url2 = $"http://{ip}/get_sensors_info?page=2"; + try + { + var url1 = $"http://{cumulus.Gw1000IpAddress}/get_sensors_info?page=1"; + var url2 = $"http://{cumulus.Gw1000IpAddress}/get_sensors_info?page=2"; - var task1 = cumulus.MyHttpClient.GetStringAsync(url1, token); - var task2 = cumulus.MyHttpClient.GetStringAsync(url2, token); + var task1 = cumulus.MyHttpClient.GetStringAsync(url1, token); + var task2 = cumulus.MyHttpClient.GetStringAsync(url2, token); - // Wait for both tasks to complete - await Task.WhenAll(task1, task2); + // Wait for both tasks to complete + await Task.WhenAll(task1, task2); - // Retrieve the results - string result1 = await task1; - string result2 = await task2; + // Retrieve the results + string result1 = await task1; + string result2 = await task2; - cumulus.LogDataMessage("GetSensorInfo: Page 1 = " + result1); - cumulus.LogDataMessage("GetSensorInfo: Page 2 = " + result2); + cumulus.LogDataMessage("GetSensorInfo: Page 1 = " + result1); + cumulus.LogDataMessage("GetSensorInfo: Page 2 = " + result2); - if (!string.IsNullOrEmpty(result1)) - { - sensors1 = result1.FromJson(); + if (!string.IsNullOrEmpty(result1)) + { + sensors1 = result1.FromJson(); + } + if (!string.IsNullOrEmpty(result2)) + { + sensors2 = result2.FromJson(); + } + + var retArr = new SensorInfo[sensors1.Length + sensors2.Length]; + sensors1.CopyTo(retArr, 0); + sensors2.CopyTo(retArr, sensors1.Length); + + return retArr; } - if (!string.IsNullOrEmpty(result2)) + catch (Exception ex) { - sensors2 = result2.FromJson(); + cumulus.LogExceptionMessage(ex, "GetSensorInfo: Error"); } - var retArr = new SensorInfo[sensors1.Length + sensors2.Length]; - sensors1.CopyTo(retArr, 0); - sensors2.CopyTo(retArr, sensors1.Length); + return null; + } + + + public void GetVersion(CancellationToken token) + { + // http://ip-address/get_version + + // response + // { + // "version": "Version: GW1100A_V2.3.4", + // "newVersion": "0", + // "platform": "ecowitt" + // } + + } + + public void GetDeviceInfo(CancellationToken token) + { + // http://ip-address/get_device_info + + //{ + // "sensorType": "1", + // "rf_freq": "1", + // "AFC": "0", + // "tz_auto": "1", + // "tz_name": "", + // "tz_index": "39", + // "dst_stat": "1", + // "radcompensation": "0", + // "date": "2024-09-06T16:36", + // "upgrade": "0", + // "apAuto": "1", + // "newVersion": "0", + // "curr_msg": "Current version:V2.3.4\r\n- Optimize RF reception performance.\r\n- Fix the issue of incorrect voltage upload for wh34/wh35/wh68 batteries.", + // "apName": "GW1100A-WIFID4D3", + // "APpwd": "", + // "time": "20" + //} + } + + public void SetDeviceInfo(CancellationToken token) + { + // http://ip-address/set_device_info + + // POST + + } + + public void GetUnits(CancellationToken token) + { + // http://ip-address/get_units_info + + // response + //{ + // "temperature": "0", 0=C 1=F + // "pressure": "0", 0=hPa 1=inHg 2=mmHg + // "wind": "2", 0=ms 1=km/h 2=mph 3=knots + // "rain": "0", 0=mm 1=in + // "light": "1" 0=kLux=? 1=W/m2 2=kfc + //} + } + + public void SetUnits(CancellationToken token) + { + // http://ip-address/set_units_info + + // POST + //{temperature: "1", pressure: "0", wind: "2", rain: "0", light: "1"} + + // response = 200 - OK + } + + + public void SetLogin(string password) + { + // http://ip-address/set_login_info + + // POST + //{ + // "pwd":"" + //} + + // Response + //{ + // "status": "1", + // "online": "0", + // "msg": "success" + //} + } + + public void GetRainTotals(CancellationToken token) + { + // http://ip-address/get_rain_totals + + // response + //{ + // "rainFallPriority": "1", 0=No Guage 1=Traditional 2=Piezo + // "list": [ + // { + // "gauge": "No rain gauge", + // "value": "0" + // }, { + // "gauge": "Traditional rain gauge", + // "value": "1" + // }, { + // "gauge": "Piezoelectric rain gauge", + // "value": "2" + // } + // ], + // "rainDay": "0.0", + // "rainWeek": "5.3", + // "rainMonth": "6.8", + // "rainYear": "572.5", + // "rainGain": "1.00", + // "rstRainDay": "0", reset hour - 0=00:00 etc + // "rstRainWeek": "1", 0=Sunday 1=Monday + // "rstRainYear":"0" reset month + //} + + // response = 200 - OK + + } + + public void SetRainTotals(CancellationToken token) + { + // http://ip-address/set_rain_totals + + // POST + //{ + // "rainDay": "0.0", + // "rainWeek": "5.3", + // "rainMonth": "6.8", + // "rainYear": "572.5", + // "rainGain": "1.01", + // "rainFallPriority": "1", + // "rstRainDay": "0", + // "rstRainWeek": "1", + // "rstRainYear": "0" + //} + + // response = 200 - OK + + } + + + public void CheckForUpgrade(CancellationToken token) + { + // http://ip-address/upgrade_process + + // POST + // {"upgrade": "check"} + + // response + //{ + // "is_new": false, + // "msg": "It's the latest version\r\nCurrent version:V2.3.4\r\n- Optimize RF reception performance.\r\n- Fix the issue of incorrect voltage upload for wh34/wh35/wh68 batteries." + //} + + } + + public void StartUpgrade(CancellationToken token) + { + // http://ip-address/upgrade_process + + // POST + // {"upgrade": "start"} + + // response + // object + // status: N 1=running + // 'over' + + } + + public void Login(string password, CancellationToken token) + { + // http://ip-address/set_login_info + + // POST + //{pwd: "base64_string"} + } + + + public void Reboot(CancellationToken token) + { + // http://ip-address/set_device_info + + // POST + // { sysreboot: 1 } - return retArr; + } + + private static string decodePassword(string base64EncodedData) + { + var base64EncodedBytes = System.Convert.FromBase64String(base64EncodedData); + return System.Text.Encoding.UTF8.GetString(base64EncodedBytes); + } + + private static string encodePassword(string plainText) + { + var plainTextBytes = System.Text.Encoding.UTF8.GetBytes(plainText); + return System.Convert.ToBase64String(plainTextBytes); } diff --git a/CumulusMX/StationSettings.cs b/CumulusMX/StationSettings.cs index 7b702b0e..25c7ec73 100644 --- a/CumulusMX/StationSettings.cs +++ b/CumulusMX/StationSettings.cs @@ -159,7 +159,8 @@ internal string GetAlpacaFormData() var ecowittHttpApi = new JsonStationSettingsHttpApi() { - ipaddress = cumulus.Gw1000IpAddress + ipaddress = cumulus.Gw1000IpAddress, + password = cumulus.EcowittHttpPassword }; var ecowitt = new JsonStationSettingsEcowitt @@ -917,6 +918,7 @@ internal string UpdateConfig(IHttpContext context) if (settings.ecowitthttpapi != null) { cumulus.Gw1000IpAddress = string.IsNullOrWhiteSpace(settings.ecowitthttpapi.ipaddress) ? null : settings.ecowitthttpapi.ipaddress.Trim(); + cumulus.EcowittHttpPassword = string.IsNullOrWhiteSpace(settings.ecowitthttpapi.password) ? null : settings.ecowitthttpapi.password.Trim(); } } catch (Exception ex) @@ -1805,6 +1807,7 @@ internal class JsonStationSettingsGw1000Conn internal class JsonStationSettingsHttpApi { public string ipaddress { get; set; } + public string password { get; set; } } internal class JsonStationSettingsEcowitt diff --git a/CumulusMX/Utils.cs b/CumulusMX/Utils.cs index 03fd86e5..9e833c13 100644 --- a/CumulusMX/Utils.cs +++ b/CumulusMX/Utils.cs @@ -94,7 +94,7 @@ public static string GetSHA256Hash(string key, string data) public static bool ValidateIPv4(string ipString) { - if (string.IsNullOrWhiteSpace(ipString)) + if (string.IsNullOrWhiteSpace(ipString) || ipString == "0.0.0.0") { return false; } diff --git a/Updates.txt b/Updates.txt index 695ec0b8..00323583 100644 --- a/Updates.txt +++ b/Updates.txt @@ -1,5 +1,17 @@ -4.2.0 - b4031 +4.2.0 - b4032 ————————————— +BETA Changes: +- Add Ecowitt HTTP API password +- Add polling time update to Ecowitt HTTP API station +- Add more Ecowitt HTTP API station error handling +- Ecowitt HTTP API station ignore apparent temp +- More HTTP API station Lightning fixes +- Davis WL Cloud station - some support for new <#LowBattList> +- Davis WLL station support for new <#LowBattList> +- Davis VP2 station support for new <#LowBattList> + + + New - New station type: Ecowitt Local HTTP API - Ecowitt Local HTTP API is an alternative the local TCP API used by the gateways and some stations @@ -14,25 +26,17 @@ New - Where is "LOW" or "0" or "1" depending on what the sensor sends Changed -- The Station Settings screen now does a two stage selection: First the manufacturer, then the station model. This shortens the long and list to select from. - +- Davis WLL/Davis Cloud stations now use the WL.com subscription level to determine if they use WL.com as a logger +- The Station Settings screen now does a two-stage selection: First the manufacturer, then the station model. This shortens the long and list to select from. +- AI2 dashboard now shows some Davis WLL hardware information Fixed -- Davis Cloud stations no longer continuosly try to fetch history data if there is no Pro subscription +- Davis Cloud stations in endless loop at startup if there is no historic data to process or access is denied +- Davis Cloud stations no longer continuously try to fetch history data if there is no Pro subscription - WMR928 Station now correctly converts indoor temperatures to the user defined units - Not logging PHP upload failures to the warning log - -4.1.4 - b4029 -————————————— -Fixed -- Davis Cloud stations in endless loop at startup if there is no historic data to process or access is denied - -Changed -- Davis WLL/DAvis Cloud stations now use the WL.com subscription level to determine if they use WL.com as a logger - - Package Updates - SQLite: Reverted to v2.1.8 pending fix from author From d9980d59b16d4da989f359d649c453e850ede5dc Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sun, 8 Sep 2024 16:03:23 +0100 Subject: [PATCH 27/63] Detect 404 errors for the Ecowitt HTTP API --- CumulusMX/EcowittHttpApiStation.cs | 183 ++++++++++++++-------------- CumulusMX/EcowittLocalApi.cs | 188 +++-------------------------- Updates.txt | 1 + 3 files changed, 110 insertions(+), 262 deletions(-) diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index 5a7724fc..13496c54 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -517,101 +517,108 @@ private async Task GetSensorIds() var sensor = sensors[i]; var name = string.Empty; - cumulus.LogDebugMessage($" - enabled={sensor.idst}, type={sensor.img}, sensor id={sensor.id}, signal={sensor.signal}, battery={sensor.batt}, name={sensor.name}"); - - // check the battery status - if (sensor.idst && sensor.signal > 0) + try { -#pragma warning disable S907 - switch (sensor.type) + cumulus.LogDebugMessage($" - enabled={sensor.idst}, type={sensor.img}, sensor id={sensor.id}, signal={sensor.signal}, battery={sensor.batt}, name={sensor.name}"); + + // check the battery status + if (sensor.idst && sensor.signal > 0) { - case 0: // wh69 - break; - case 1: // wh68 - name = "wh68"; - goto case 1003; - case 2: // wh80 - name = "wh80"; - // if a WS80 is connected, it has a 4.75 second update rate, so reduce the MX update rate from the default 10 seconds - if (updateRate > 4000 && updateRate != 4000) - { - cumulus.LogMessage($"GetSensorIds: WS80 sensor detected, changing the update rate from {(updateRate / 1000):D} seconds to 4 seconds"); - updateRate = 4000; - } - goto case 1003; - case 3: // wh40 - name = "wh40"; - goto case 1003; - case 4: // wh25 - name = "wh25"; - goto case 1003; - case 5: // wh26 - name = "wh326"; - goto case 1001; - case int n when (n > 5 && n < 14): // wh31 - T&H (8 chan) - name = "wh31ch" + (sensor.type - 5); - goto case 1001; - case int n when (n > 13 && n < 22): // wh51 - soil moisture (8 chan) - break; - case int n when (n > 21 && n < 26): // wh41 - pm2.5 (4 chan) - name = "wh41ch" + (sensor.type - 21); - goto case 1003; - case 26: // wh57 - lightning - name = "wh57"; - goto case 1003; - case int n when (n > 26 && n < 31): // wh55 - leak (4 chan) - name = "wh55ch" + (sensor.type - 26); - goto case 1003; - case int n when (n > 30 && n < 39): // wh34 - Temp (8 chan) - name = "wh34ch" + (sensor.type - 30); - goto case 1003; - case 39: // wh45 - co2 - name = "wh45"; - goto case 1003; - case int n when (n > 39 && n < 48): // wh35 - leasf wet (8 chan) - name = "wh35ch" + (sensor.type - 39); - goto case 1003; - case 48: // wh90 - name = "wh90"; - // if a WS90 is connected, it has a 8.8 second update rate, so reduce the MX update rate from the default 10 seconds - if (updateRate > 8000 && updateRate != 8000) - { - cumulus.LogMessage($"GetSensorIds: WS90 sensor detected, changing the update rate from {(updateRate / 1000):D} seconds to 8 seconds"); - updateRate = 8000; - } - goto case 1003; - case 49: // wh85 - name = "wh85"; - // if a WH85 is connected, it has a 8.5 second update rate, so reduce the MX update rate from the default 10 seconds - if (updateRate > 8000 && updateRate != 8000) - { - cumulus.LogMessage($"GetSensorIds: WH85 sensor detected, changing the update rate from {(updateRate / 1000):D} seconds to 8 seconds"); - updateRate = 8000; - } - goto case 1003; +#pragma warning disable S907 + switch (sensor.type) + { + case 0: // wh69 + break; + case 1: // wh68 + name = "wh68"; + goto case 1003; + case 2: // wh80 + name = "wh80"; + // if a WS80 is connected, it has a 4.75 second update rate, so reduce the MX update rate from the default 10 seconds + if (updateRate > 4000 && updateRate != 4000) + { + cumulus.LogMessage($"GetSensorIds: WS80 sensor detected, changing the update rate from {(updateRate / 1000):D} seconds to 4 seconds"); + updateRate = 4000; + } + goto case 1003; + case 3: // wh40 + name = "wh40"; + goto case 1003; + case 4: // wh25 + name = "wh25"; + goto case 1003; + case 5: // wh26 + name = "wh326"; + goto case 1001; + case int n when (n > 5 && n < 14): // wh31 - T&H (8 chan) + name = "wh31ch" + (sensor.type - 5); + goto case 1001; + case int n when (n > 13 && n < 22): // wh51 - soil moisture (8 chan) + break; + case int n when (n > 21 && n < 26): // wh41 - pm2.5 (4 chan) + name = "wh41ch" + (sensor.type - 21); + goto case 1003; + case 26: // wh57 - lightning + name = "wh57"; + goto case 1003; + case int n when (n > 26 && n < 31): // wh55 - leak (4 chan) + name = "wh55ch" + (sensor.type - 26); + goto case 1003; + case int n when (n > 30 && n < 39): // wh34 - Temp (8 chan) + name = "wh34ch" + (sensor.type - 30); + goto case 1003; + case 39: // wh45 - co2 + name = "wh45"; + goto case 1003; + case int n when (n > 39 && n < 48): // wh35 - leasf wet (8 chan) + name = "wh35ch" + (sensor.type - 39); + goto case 1003; + case 48: // wh90 + name = "wh90"; + // if a WS90 is connected, it has a 8.8 second update rate, so reduce the MX update rate from the default 10 seconds + if (updateRate > 8000 && updateRate != 8000) + { + cumulus.LogMessage($"GetSensorIds: WS90 sensor detected, changing the update rate from {(updateRate / 1000):D} seconds to 8 seconds"); + updateRate = 8000; + } + goto case 1003; + case 49: // wh85 + name = "wh85"; + // if a WH85 is connected, it has a 8.5 second update rate, so reduce the MX update rate from the default 10 seconds + if (updateRate > 8000 && updateRate != 8000) + { + cumulus.LogMessage($"GetSensorIds: WH85 sensor detected, changing the update rate from {(updateRate / 1000):D} seconds to 8 seconds"); + updateRate = 8000; + } + goto case 1003; - case 1001: // battery type 1 (0=OK, 1=LOW) - if (sensor.batt == 1) - { - batteryLow = true; - LowBatteryDevices.Add(name + " - LOW"); - } - break; + case 1001: // battery type 1 (0=OK, 1=LOW) + if (sensor.batt == 1) + { + batteryLow = true; + LowBatteryDevices.Add(name + " - LOW"); + } + break; - case 1003: // battery type 3 (0-1=LOW, 2-5=OK, 6=DC, 9=OFF) - if (!TestBattery3(sensor.batt)) - { - batteryLow = true; - LowBatteryDevices.Add(name + " - " + sensor.batt); - } - break; + case 1003: // battery type 3 (0-1=LOW, 2-5=OK, 6=DC, 9=OFF) + if (!TestBattery3(sensor.batt)) + { + batteryLow = true; + LowBatteryDevices.Add(name + " - " + sensor.batt); + } + break; - default: - cumulus.LogWarningMessage($"Unknown sensor type in SendorIds. Model={sensor.img}, type={sensor.type}"); - break; - } + default: + cumulus.LogWarningMessage($"Unknown sensor type in SendorIds. Model={sensor.img}, type={sensor.type}"); + break; + } #pragma warning restore S907 + } + } + catch (Exception ex) + { + cumulus.LogExceptionMessage(ex, $"GetSensorIds: Error processing sensor[{i}]"); } } } diff --git a/CumulusMX/EcowittLocalApi.cs b/CumulusMX/EcowittLocalApi.cs index 42c6f372..b997c7e0 100644 --- a/CumulusMX/EcowittLocalApi.cs +++ b/CumulusMX/EcowittLocalApi.cs @@ -192,6 +192,13 @@ public LiveData GetLiveData(CancellationToken token) return json; } } + catch (System.Net.Http.HttpRequestException ex) + { + if (ex.StatusCode == System.Net.HttpStatusCode.NotFound) + { + cumulus.LogErrorMessage("GetLiveData: Error - This Station does not support the HTTP API!"); + } + } catch (Exception ex) { cumulus.LogExceptionMessage(ex, "GetLiveData: Error"); @@ -207,180 +214,6 @@ public async Task GetSensorInfo(CancellationToken token) // http://ip-address/get_sensors_info?page=1 // http://ip-address/get_sensors_info?page=2 - // Example of page=1 - /* - - [ - { - "img": "wh85", - "type": "49", - "name": "Wind & Rain", - "id": "FFFFFFFF", - "batt": "9", - "signal": "0", - "idst": "1" - }, - { - "img": "wh90", - "type": "48", - "name": "Temp & Humidity & Solar & Wind & Rain", - "id": "FFFFFFFE", - "batt": "9", - "signal": "0", - "idst": "0" - }, - { - "img": "wh69", - "type": "0", - "name": "Temp & Humidity & Solar & Wind & Rain", - "id": "FFFFFFFE", - "batt": "9", - "signal": "0", - "idst": "0" - }, - { - "img": "wh68", - "type": "1", - "name": "Solar & Wind", - "id": "FFFFFFFE", - "batt": "9", - "signal": "0", - "idst": "0" - }, - { - "img": "wh40", - "type": "3", - "name": "Rain", - "id": "E1EF", - "batt": "0", - "signal": "3", - "idst": "1" - }, - { - "img": "wh25", - "type": "4", - "name": "Temp & Humidity & Pressure", - "id": "FFFFFFFF", - "batt": "9", - "signal": "0", - "idst": "1" - }, - { - "img": "wh26", - "type": "5", - "name": "Temp & Humidity", - "id": "FFFFFFFE", - "batt": "9", - "signal": "0", - "idst": "0" - }, - { - "img": "wh80", - "type": "2", - "name": "Temp & Humidity & Solar & Wind", - "id": "E0038", - "batt": "5", - "signal": "4", - "idst": "1" - }, - { - "img": "wh57", - "type": "26", - "name": "Lightning", - "id": "FFFFFFFE", - "batt": "9", - "signal": "0", - "idst": "0" - }, - { - "img": "wh45", - "type": "39", - "name": "PM25 & PM10 & CO2", - "id": "FFFFFFFE", - "batt": "9", - "signal": "0", - "idst": "0" - }, - { - "img": "wh41", - "type": "22", - "name": "PM2.5 CH1", - "id": "34", - "batt": "9", - "signal": "0", - "idst": "1" - }, - .. chans 2-4 - { - "img": "wh55", - "type": "27", - "name": "Leak CH1", - "id": "FFFFFFFE", - "batt": "9", - "signal": "0", - "idst": "0" - }, - .. chans 2-4 - { - "img": "wh31", - "type": "6", - "name": "Temp & Humidity CH1", - "id": "B0", - "batt": "0", - "signal": "4", - "idst": "1" - }, - .. chans 2-7 - ] - */ - - - // Example page=2 - /* - [ - { - "img": "wh31", - "type": "13", - "name": "Temp & Humidity CH8", - "id": "FFFFFFFE", - "batt": "9", - "signal": "0", - "idst": "0" - }, - { - "img": "wh51", - "type": "14", - "name": "Soil moisture CH1", - "id": "FFFFFFFE", - "batt": "9", - "signal": "0", - "idst": "0" - }, - .. chans 2-8 - { - "img": "wh34", - "type": "31", - "name": "Temp CH1", - "id": "FFFFFFFE", - "batt": "9", - "signal": "0", - "idst": "0" - }, - .. chans 2-8 - { - "img": "wh35", - "type": "40", - "name": "Leaf Wetness CH1", - "id": "FFFFFFFE", - "batt": "9", - "signal": "0", - "idst": "0" - }, - .. chans 2-8 - ] - - */ - if (!Utils.ValidateIPv4(cumulus.Gw1000IpAddress)) { cumulus.LogErrorMessage("GetSensorInfo: Invalid station IP address: " + cumulus.Gw1000IpAddress); @@ -424,6 +257,13 @@ .. chans 2-8 return retArr; } + catch (System.Net.Http.HttpRequestException ex) + { + if (ex.StatusCode == System.Net.HttpStatusCode.NotFound) + { + cumulus.LogErrorMessage("GetSensorInfo: Error - This Station does not support the HTTP API!"); + } + } catch (Exception ex) { cumulus.LogExceptionMessage(ex, "GetSensorInfo: Error"); diff --git a/Updates.txt b/Updates.txt index 00323583..65163593 100644 --- a/Updates.txt +++ b/Updates.txt @@ -9,6 +9,7 @@ BETA Changes: - Davis WL Cloud station - some support for new <#LowBattList> - Davis WLL station support for new <#LowBattList> - Davis VP2 station support for new <#LowBattList> +- Detect 404 errors for the Ecowitt HTTP API From ad68fc5773071c56c48174582b4d5adbb5347855 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sun, 8 Sep 2024 16:59:10 +0100 Subject: [PATCH 28/63] External Programs now set the working directory to the location of the executable/script External programs do a file exists check --- CumulusMX/Alarm.cs | 107 +++++++++++++++++++------------ CumulusMX/AlarmsUser.cs | 36 +++++++---- CumulusMX/Cumulus.cs | 124 ++++++++++++++++++++++-------------- CumulusMX/Utils.cs | 27 ++++---- CumulusMX/WeatherStation.cs | 31 +++++---- Updates.txt | 2 + 6 files changed, 200 insertions(+), 127 deletions(-) diff --git a/CumulusMX/Alarm.cs b/CumulusMX/Alarm.cs index ed1733ce..1ba24f23 100644 --- a/CumulusMX/Alarm.cs +++ b/CumulusMX/Alarm.cs @@ -1,4 +1,5 @@ using System; +using System.IO; using System.Runtime.Serialization; using System.Threading.Tasks; @@ -118,24 +119,31 @@ private void DoTriggered(bool value) if (!string.IsNullOrEmpty(Action)) { - try + if (!File.Exists(Action)) { - var args = string.Empty; - // Prepare the process to run - if (!string.IsNullOrEmpty(ActionParams)) + cumulus.LogWarningMessage($"Warning: Alarm ({Name}): External program: '{Action}' does not exist"); + } + else + { + try { - var parser = new TokenParser(cumulus.TokenParserOnToken) + var args = string.Empty; + // Prepare the process to run + if (!string.IsNullOrEmpty(ActionParams)) { - InputText = ActionParams - }; - args = parser.ToStringFromString(); + var parser = new TokenParser(cumulus.TokenParserOnToken) + { + InputText = ActionParams + }; + args = parser.ToStringFromString(); + } + cumulus.LogMessage($"Alarm ({Name}): Starting external program: '{Action}', with parameters: {args}"); + Utils.RunExternalTask(Action, args, false, false, ShowWindow); + } + catch (Exception ex) + { + cumulus.LogErrorMessage($"Alarm ({Name}): Error executing external program '{Action}': {ex.Message}"); } - cumulus.LogMessage($"Alarm ({Name}): Starting external program: '{Action}', with parameters: {args}"); - Utils.RunExternalTask(Action, args, false, false, ShowWindow); - } - catch (Exception ex) - { - cumulus.LogErrorMessage($"Alarm ({Name}): Error executing external program '{Action}': {ex.Message}"); } } } @@ -292,24 +300,31 @@ private void DoUpTriggered(bool value) if (!string.IsNullOrEmpty(Action)) { - try + if (!File.Exists(Action)) { - var args = string.Empty; - // Prepare the process to run - if (!string.IsNullOrEmpty(ActionParams)) + cumulus.LogWarningMessage($"Warning: Alarm ({Name}): External program: '{Action}' does not exist"); + } + else + { + try { - var parser = new TokenParser(cumulus.TokenParserOnToken) + var args = string.Empty; + // Prepare the process to run + if (!string.IsNullOrEmpty(ActionParams)) { - InputText = ActionParams - }; - args = parser.ToStringFromString(); + var parser = new TokenParser(cumulus.TokenParserOnToken) + { + InputText = ActionParams + }; + args = parser.ToStringFromString(); + } + cumulus.LogMessage($"Alarm ({NameUp}): Starting external program: '{Action}', with parameters: {args}"); + Utils.RunExternalTask(Action, args, false, false, ShowWindow); + } + catch (Exception ex) + { + cumulus.LogErrorMessage($"Alarm ({NameUp}): Error executing external program '{Action}': {ex.Message}"); } - cumulus.LogMessage($"Alarm ({NameUp}): Starting external program: '{Action}', with parameters: {args}"); - Utils.RunExternalTask(Action, args, false, false, ShowWindow); - } - catch (Exception ex) - { - cumulus.LogErrorMessage($"Alarm ({NameUp}): Error executing external program '{Action}': {ex.Message}"); } } } @@ -372,24 +387,32 @@ private void DoDownTriggered(bool value) if (!string.IsNullOrEmpty(Action)) { - try + if (!File.Exists(Action)) + { + cumulus.LogWarningMessage($"Warning: Alarm ({Name}): External program: '{Action}' does not exist"); + } + else { - var args = string.Empty; - // Prepare the process to run - if (!string.IsNullOrEmpty(ActionParams)) + + try { - var parser = new TokenParser(cumulus.TokenParserOnToken) + var args = string.Empty; + // Prepare the process to run + if (!string.IsNullOrEmpty(ActionParams)) { - InputText = ActionParams - }; - args = parser.ToStringFromString(); + var parser = new TokenParser(cumulus.TokenParserOnToken) + { + InputText = ActionParams + }; + args = parser.ToStringFromString(); + } + cumulus.LogMessage($"Alarm ({NameDown}): Starting external program: '{Action}', with parameters: {args}"); + Utils.RunExternalTask(Action, args, false, false, ShowWindow); + } + catch (Exception ex) + { + cumulus.LogErrorMessage($"Alarm ({NameDown}): Error executing external program '{Action}': {ex.Message}"); } - cumulus.LogMessage($"Alarm ({NameDown}): Starting external program: '{Action}', with parameters: {args}"); - Utils.RunExternalTask(Action, args, false, false, ShowWindow); - } - catch (Exception ex) - { - cumulus.LogErrorMessage($"Alarm ({NameDown}): Error executing external program '{Action}': {ex.Message}"); } } } diff --git a/CumulusMX/AlarmsUser.cs b/CumulusMX/AlarmsUser.cs index fcef8d64..800dc2e9 100644 --- a/CumulusMX/AlarmsUser.cs +++ b/CumulusMX/AlarmsUser.cs @@ -1,4 +1,5 @@ using System; +using System.IO; using System.Runtime.Serialization; using System.Threading.Tasks; @@ -154,24 +155,31 @@ private void doTriggered(bool value) if (!string.IsNullOrEmpty(Action)) { - try + if (!File.Exists(Action)) { - var args = string.Empty; - // Prepare the process to run - if (!string.IsNullOrEmpty(ActionParams)) + cumulus.LogWarningMessage($"Warning: Alarm ({Name}): External program: '{Action}' does not exist"); + } + else + { + try { - var parser = new TokenParser(cumulus.TokenParserOnToken) + var args = string.Empty; + // Prepare the process to run + if (!string.IsNullOrEmpty(ActionParams)) { - InputText = ActionParams - }; - args = parser.ToStringFromString(); + var parser = new TokenParser(cumulus.TokenParserOnToken) + { + InputText = ActionParams + }; + args = parser.ToStringFromString(); + } + cumulus.LogMessage($"User Alarm ({Name}): Starting external program: '{Action}', with parameters: {args}"); + Utils.RunExternalTask(Action, args, false, false, ShowWindow); + } + catch (Exception ex) + { + cumulus.LogErrorMessage($"User Alarm ({Name}): Error executing external program '{Action}': {ex.Message}"); } - cumulus.LogMessage($"User Alarm ({Name}): Starting external program: '{Action}', with parameters: {args}"); - Utils.RunExternalTask(Action, args, false, false, ShowWindow); - } - catch (Exception ex) - { - cumulus.LogErrorMessage($"User Alarm ({Name}): Error executing external program '{Action}': {ex.Message}"); } } } diff --git a/CumulusMX/Cumulus.cs b/CumulusMX/Cumulus.cs index 53db9bf1..764c4ef0 100644 --- a/CumulusMX/Cumulus.cs +++ b/CumulusMX/Cumulus.cs @@ -1118,14 +1118,21 @@ public void Initialise(int HTTPport, bool DebugEnabled, string startParms) // do we have a start-up task to run? if (!string.IsNullOrEmpty(ProgramOptions.StartupTask)) { - LogMessage($"Running start-up task: {ProgramOptions.StartupTask}, arguments: {ProgramOptions.StartupTaskParams}, wait: {ProgramOptions.StartupTaskWait}"); - try + if (!File.Exists(ProgramOptions.StartupTask)) { - Utils.RunExternalTask(ProgramOptions.StartupTask, ProgramOptions.StartupTaskParams, ProgramOptions.StartupTaskWait); + LogWarningMessage($"Waring: Start-up task: '{ProgramOptions.StartupTask}' does not exist"); } - catch (Exception ex) + else { - LogErrorMessage($"Error running start-up task: {ex.Message}"); + LogMessage($"Running start-up task: {ProgramOptions.StartupTask}, arguments: {ProgramOptions.StartupTaskParams}, wait: {ProgramOptions.StartupTaskWait}"); + try + { + Utils.RunExternalTask(ProgramOptions.StartupTask, ProgramOptions.StartupTaskParams, ProgramOptions.StartupTaskWait); + } + catch (Exception ex) + { + LogErrorMessage($"Error running start-up task: {ex.Message}"); + } } } @@ -2306,24 +2313,31 @@ internal void RealtimeTimerTick(object sender, ElapsedEventArgs elapsedEventArgs if (!string.IsNullOrEmpty(RealtimeProgram)) { - try + if (!File.Exists(ProgramOptions.StartupTask)) { - var args = string.Empty; - - if (!string.IsNullOrEmpty(RealtimeParams)) + LogWarningMessage($"Warning: Realtime program '{RealtimeProgram}' does not exist"); + } + else + { + try { - var parser = new TokenParser(TokenParserOnToken) + var args = string.Empty; + + if (!string.IsNullOrEmpty(RealtimeParams)) { - InputText = RealtimeParams - }; - args = parser.ToStringFromString(); + var parser = new TokenParser(TokenParserOnToken) + { + InputText = RealtimeParams + }; + args = parser.ToStringFromString(); + } + LogDebugMessage($"Realtime[{cycle}]: Execute realtime program - {RealtimeProgram}, with parameters - {args}"); + Utils.RunExternalTask(RealtimeProgram, args, false); + } + catch (Exception ex) + { + LogErrorMessage($"Realtime[{cycle}]: Error in realtime program - {RealtimeProgram}. Error: {ex.Message}"); } - LogDebugMessage($"Realtime[{cycle}]: Execute realtime program - {RealtimeProgram}, with parameters - {args}"); - Utils.RunExternalTask(RealtimeProgram, args, false); - } - catch (Exception ex) - { - LogErrorMessage($"Realtime[{cycle}]: Error in realtime program - {RealtimeProgram}. Error: {ex.Message}"); } } } @@ -8584,24 +8598,31 @@ public void Stop() // do we have a shutdown task to run? if (!string.IsNullOrEmpty(ProgramOptions.ShutdownTask)) { - try + if (!File.Exists(ProgramOptions.ShutdownTask)) { - var args = string.Empty; - - if (!string.IsNullOrEmpty(ProgramOptions.ShutdownTaskParams)) + LogWarningMessage($"Warning: Shutdown eextrernal task: '{ProgramOptions.ShutdownTask}' does not exist"); + } + else + { + try { - var parser = new TokenParser(TokenParserOnToken) + var args = string.Empty; + + if (!string.IsNullOrEmpty(ProgramOptions.ShutdownTaskParams)) { - InputText = ProgramOptions.ShutdownTaskParams - }; - args = parser.ToStringFromString(); + var parser = new TokenParser(TokenParserOnToken) + { + InputText = ProgramOptions.ShutdownTaskParams + }; + args = parser.ToStringFromString(); + } + LogMessage($"Running shutdown task: {ProgramOptions.ShutdownTask}, arguments: {args}"); + Utils.RunExternalTask(ProgramOptions.ShutdownTask, args, false); + } + catch (Exception ex) + { + LogMessage($"Error running shutdown task: {ex.Message}"); } - LogMessage($"Running shutdown task: {ProgramOptions.ShutdownTask}, arguments: {args}"); - Utils.RunExternalTask(ProgramOptions.ShutdownTask, args, false); - } - catch (Exception ex) - { - LogMessage($"Error running shutdown task: {ex.Message}"); } } @@ -8701,25 +8722,32 @@ public async Task DoHTMLFiles() if (!string.IsNullOrEmpty(ExternalProgram)) { - try + if (!File.Exists(ExternalProgram)) { - var args = string.Empty; - - if (!string.IsNullOrEmpty(ExternalParams)) + LogWarningMessage($"Warning: Interval: External program '{ExternalProgram}' does not exist"); + } + else + { + try { - var parser = new TokenParser(TokenParserOnToken) + var args = string.Empty; + + if (!string.IsNullOrEmpty(ExternalParams)) { - InputText = ExternalParams - }; - args = parser.ToStringFromString(); + var parser = new TokenParser(TokenParserOnToken) + { + InputText = ExternalParams + }; + args = parser.ToStringFromString(); + } + LogDebugMessage("Interval: Executing external program " + ExternalProgram + " " + args); + Utils.RunExternalTask(ExternalProgram, args, false); + LogDebugMessage("Interval: External program started"); + } + catch (Exception ex) + { + LogWarningMessage("Interval: Error starting external program: " + ex.Message); } - LogDebugMessage("Interval: Executing program " + ExternalProgram + " " + args); - Utils.RunExternalTask(ExternalProgram, args, false); - LogDebugMessage("Interval: External program started"); - } - catch (Exception ex) - { - LogWarningMessage("Interval: Error starting external program: " + ex.Message); } } diff --git a/CumulusMX/Utils.cs b/CumulusMX/Utils.cs index 9e833c13..673d7381 100644 --- a/CumulusMX/Utils.cs +++ b/CumulusMX/Utils.cs @@ -228,18 +228,23 @@ public static string ExceptionToString(Exception ex, out string message) public static void RunExternalTask(string task, string parameters, bool wait, bool redirectError = false, bool createwindow = false) { - var process = new System.Diagnostics.Process(); - process.StartInfo.FileName = task; - process.StartInfo.Arguments = parameters; - process.StartInfo.UseShellExecute = false; - process.StartInfo.RedirectStandardError = redirectError; - process.StartInfo.WindowStyle = createwindow ? System.Diagnostics.ProcessWindowStyle.Normal : System.Diagnostics.ProcessWindowStyle.Hidden; - process.StartInfo.CreateNoWindow = !createwindow; - process.Start(); - - if (wait) + var file = new FileInfo(task); + if (file.Exists) { - process.WaitForExit(); + using var process = new System.Diagnostics.Process(); + process.StartInfo.FileName = file.FullName; + process.StartInfo.Arguments = parameters; + process.StartInfo.WorkingDirectory = file.DirectoryName; + process.StartInfo.UseShellExecute = false; + process.StartInfo.RedirectStandardError = redirectError; + process.StartInfo.WindowStyle = createwindow ? System.Diagnostics.ProcessWindowStyle.Normal : System.Diagnostics.ProcessWindowStyle.Hidden; + process.StartInfo.CreateNoWindow = !createwindow; + process.Start(); + + if (wait) + { + process.WaitForExit(); + } } } diff --git a/CumulusMX/WeatherStation.cs b/CumulusMX/WeatherStation.cs index 2bb39d47..c4d12786 100644 --- a/CumulusMX/WeatherStation.cs +++ b/CumulusMX/WeatherStation.cs @@ -7578,22 +7578,29 @@ public void DayReset(DateTime timestamp) if (!string.IsNullOrEmpty(cumulus.DailyProgram)) { - try + if (!File.Exists(cumulus.DailyProgram)) + { + cumulus.LogWarningMessage($"Warning: Daily external program '{cumulus.DailyProgram}' does not exist"); + } + else { - // Prepare the process to run - var args = string.Empty; + try + { + // Prepare the process to run + var args = string.Empty; - if (!string.IsNullOrEmpty(cumulus.DailyParams)) + if (!string.IsNullOrEmpty(cumulus.DailyParams)) + { + var parser = new TokenParser(cumulus.TokenParserOnToken) { InputText = cumulus.DailyParams }; + args = parser.ToStringFromString(); + } + cumulus.LogMessage("Executing daily program: " + cumulus.DailyProgram + " params: " + args); + Utils.RunExternalTask(cumulus.DailyProgram, args, false); + } + catch (Exception ex) { - var parser = new TokenParser(cumulus.TokenParserOnToken) { InputText = cumulus.DailyParams }; - args = parser.ToStringFromString(); + cumulus.LogErrorMessage("Error executing external program: " + ex.Message); } - cumulus.LogMessage("Executing daily program: " + cumulus.DailyProgram + " params: " + args); - Utils.RunExternalTask(cumulus.DailyProgram, args, false); - } - catch (Exception ex) - { - cumulus.LogErrorMessage("Error executing external program: " + ex.Message); } } diff --git a/Updates.txt b/Updates.txt index 65163593..a53e6b38 100644 --- a/Updates.txt +++ b/Updates.txt @@ -10,6 +10,7 @@ BETA Changes: - Davis WLL station support for new <#LowBattList> - Davis VP2 station support for new <#LowBattList> - Detect 404 errors for the Ecowitt HTTP API +- External Programs now set the working directory to the location of the executable/script rather than the Cumulus MX home directory @@ -30,6 +31,7 @@ Changed - Davis WLL/Davis Cloud stations now use the WL.com subscription level to determine if they use WL.com as a logger - The Station Settings screen now does a two-stage selection: First the manufacturer, then the station model. This shortens the long and list to select from. - AI2 dashboard now shows some Davis WLL hardware information +- External Programs now sets the working directory to the location of the executable/script rather than the Cumulus MX home directory Fixed - Davis Cloud stations in endless loop at startup if there is no historic data to process or access is denied From e1db3e595e28e02d8a4bf7c8ada14893861fa6c9 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sun, 8 Sep 2024 17:14:56 +0100 Subject: [PATCH 29/63] Allow for Ecowitt Lightning sensor factory reset/new device --- CumulusMX/EcowittHttpApiStation.cs | 34 +++++++++++++++++------------- Updates.txt | 2 +- 2 files changed, 20 insertions(+), 16 deletions(-) diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index 13496c54..b11e246c 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -1023,7 +1023,7 @@ private void ProcessLightning(EcowittLocalApi.LightningSensor[] sensors, DateTim if (sensor.distanceVal.HasValue && sensor.distanceUnit != null) { // Sends a default value of 255km until the first strike is detected - if (sensor.distanceVal.Value > 254.9) + if (sensor.distanceVal.Value > 254.9 ) { newLightningDistance = 999; } @@ -1032,31 +1032,35 @@ private void ProcessLightning(EcowittLocalApi.LightningSensor[] sensors, DateTim newLightningDistance = ConvertUnits.KmtoUserUnits(sensor.distanceVal.Value); } } + else if (sensor.distance.Contains("--")) + { + newLightningDistance = 999; + } //Lightning time (UTC) if (!string.IsNullOrEmpty(sensor.timestamp)) { - // oh my god, it sends the time as "MM/dd/yyyy HH: mm: ss" for some locales - // TODO: what is default time if not strikes detected yet? - var arr = sensor.timestamp.Replace(": ", ":").Split(' '); - var date = arr[0].Split('/'); - var time = arr[1].Split(':'); - - //if (sensor.timestamp == "default string") - //{ - // newLightningTime = new DateTime(1900, 1, 1, 0, 0, 0, 0, DateTimeKind.Utc); - //} - //else - //{ + if (sensor.timestamp.Contains("--")) + { + newLightningTime = new DateTime(1900, 1, 1, 0, 0, 0, 0, DateTimeKind.Utc); + } + else + { + // oh my god, it sends the time as "MM/dd/yyyy HH: mm: ss" for some locales + // TODO: what is default time if not strikes detected yet? + var arr = sensor.timestamp.Replace(": ", ":").Split(' '); + var date = arr[0].Split('/'); + var time = arr[1].Split(':'); + newLightningTime = new DateTime( int.Parse(date[2]), int.Parse(date[0]), - int.Parse(date[1]), + int.Parse(date[1]), int.Parse(time[0]), int.Parse(time[1]), int.Parse(time[2]), 0, DateTimeKind.Utc); - //} + } } //Lightning strikes today diff --git a/Updates.txt b/Updates.txt index a53e6b38..7a565eea 100644 --- a/Updates.txt +++ b/Updates.txt @@ -11,7 +11,7 @@ BETA Changes: - Davis VP2 station support for new <#LowBattList> - Detect 404 errors for the Ecowitt HTTP API - External Programs now set the working directory to the location of the executable/script rather than the Cumulus MX home directory - +- Allow for Ecowitt Lightning sensor factory reset/new device New From b11167ef7c23babbf276c59457ea9f65d09b2577 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Mon, 9 Sep 2024 16:57:12 +0100 Subject: [PATCH 30/63] Make cursor visible in console after exit - no idea what is hiding it! --- CumulusMX/Program.cs | 8 ++++++++ Updates.txt | 7 +++++-- 2 files changed, 13 insertions(+), 2 deletions(-) diff --git a/CumulusMX/Program.cs b/CumulusMX/Program.cs index cc5fea54..0ac65ec4 100644 --- a/CumulusMX/Program.cs +++ b/CumulusMX/Program.cs @@ -93,6 +93,8 @@ private static async Task Main(string[] args) Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff ") + "Cumulus terminating"); svcTextListener.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff ") + "Cumulus terminating"); } + + Console.CursorVisible = true; }; @@ -114,6 +116,7 @@ private static async Task Main(string[] args) Trace.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff ") + "Cumulus has shutdown"); ev.Cancel = true; exitSystem = true; + Console.CursorVisible = true; }; AppDomain.CurrentDomain.UnhandledException += UnhandledExceptionTrapper; @@ -302,6 +305,8 @@ private static async Task Main(string[] args) { await Task.Delay(500); } + + Console.CursorVisible = true; } private static void Usage() @@ -316,6 +321,7 @@ private static void Usage() Console.WriteLine(" -user - Specifies the user to run the service under (Linux only)"); Console.WriteLine(" -service - Must be used when running as service (Linux only)"); Console.WriteLine("\nCumulus terminating"); + Console.CursorVisible = true; Environment.Exit(1); } @@ -336,6 +342,7 @@ private static void UnhandledExceptionTrapper(object sender, UnhandledExceptionE { try { + Console.CursorVisible = true; Trace.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff ") + "!!! Unhandled Exception !!!"); Trace.WriteLine(e.ExceptionObject.ToString()); @@ -352,6 +359,7 @@ private static void UnhandledExceptionTrapper(object sender, UnhandledExceptionE Console.WriteLine("Press Enter to terminate"); Console.ReadLine(); } + Thread.Sleep(1000); Environment.Exit(1); } diff --git a/Updates.txt b/Updates.txt index 7a565eea..69c87fda 100644 --- a/Updates.txt +++ b/Updates.txt @@ -1,6 +1,7 @@ -4.2.0 - b4032 +4.2.0 - b4033 ————————————— -BETA Changes: +BETA b4033 Changes: +——————————————————— - Add Ecowitt HTTP API password - Add polling time update to Ecowitt HTTP API station - Add more Ecowitt HTTP API station error handling @@ -12,6 +13,7 @@ BETA Changes: - Detect 404 errors for the Ecowitt HTTP API - External Programs now set the working directory to the location of the executable/script rather than the Cumulus MX home directory - Allow for Ecowitt Lightning sensor factory reset/new device +- Latest AI2 updates applied New @@ -32,6 +34,7 @@ Changed - The Station Settings screen now does a two-stage selection: First the manufacturer, then the station model. This shortens the long and list to select from. - AI2 dashboard now shows some Davis WLL hardware information - External Programs now sets the working directory to the location of the executable/script rather than the Cumulus MX home directory +- Latest AI2 updates applied Fixed - Davis Cloud stations in endless loop at startup if there is no historic data to process or access is denied From a371b2ef04629ee1a851d0636b71e81af53a0eaa Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Tue, 10 Sep 2024 11:09:42 +0100 Subject: [PATCH 31/63] Improved Ecowitt HTTP API station error handling/recovery when station goes offline --- CumulusMX/CumulusMX.csproj | 4 +- CumulusMX/EcowittHttpApiStation.cs | 261 +++++++++++++++-------------- CumulusMX/EcowittLocalApi.cs | 10 +- Updates.txt | 13 +- 4 files changed, 152 insertions(+), 136 deletions(-) diff --git a/CumulusMX/CumulusMX.csproj b/CumulusMX/CumulusMX.csproj index 8ae6b6e7..fea08e75 100644 --- a/CumulusMX/CumulusMX.csproj +++ b/CumulusMX/CumulusMX.csproj @@ -55,7 +55,7 @@ - 4.2.0.4033-beta + 4.2.0.4034-beta Copyright © 2015-$([System.DateTime]::Now.ToString('yyyy')) Cumulus MX $(AssemblyName) @@ -98,7 +98,7 @@ - + diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index b11e246c..450a4796 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -145,175 +145,178 @@ public override void Start() try { - DateTime dataLastRead; + DateTime dataLastRead = DateTime.Now; double delay; while (!cumulus.cancellationToken.IsCancellationRequested) { var rawData = localApi.GetLiveData(cumulus.cancellationToken); - dataLastRead = DateTime.Now; - - // process the common_list sensors - ProcessCommonList(rawData.common_list, dataLastRead); - - // process base station values - ProcessWh25(rawData.wh25, dataLastRead); - - // process rain values - if (cumulus.Gw1000PrimaryRainSensor == 0 && rawData.rain != null) - { - ProcessRain(rawData.rain); - } - else if (cumulus.Gw1000PrimaryRainSensor == 1 && rawData.piezoRain != null) + if (rawData is not null) { - ProcessRain(rawData.piezoRain); - } + dataLastRead = DateTime.Now; - if (rawData.lightning != null) - { - ProcessLightning(rawData.lightning, dataLastRead); - } + // process the common_list sensors + ProcessCommonList(rawData.common_list, dataLastRead); - if (rawData.co2 != null) - { - ProcessCo2(rawData.co2); - } + // process base station values + ProcessWh25(rawData.wh25, dataLastRead); - if (rawData.ch_pm25 != null) - { - ProcessChPm25(rawData.ch_pm25); - } + // process rain values + if (cumulus.Gw1000PrimaryRainSensor == 0 && rawData.rain != null) + { + ProcessRain(rawData.rain); + } + else if (cumulus.Gw1000PrimaryRainSensor == 1 && rawData.piezoRain != null) + { + ProcessRain(rawData.piezoRain); + } - if (rawData.ch_leak != null) - { - ProcessLeak(rawData.ch_leak); - } + if (rawData.lightning != null) + { + ProcessLightning(rawData.lightning, dataLastRead); + } - if (rawData.ch_aisle != null) - { - ProcessExtraTempHum(rawData.ch_aisle, dataLastRead); - } + if (rawData.co2 != null) + { + ProcessCo2(rawData.co2); + } - if (rawData.ch_temp != null) - { - ProcessUserTemp(rawData.ch_temp); - } + if (rawData.ch_pm25 != null) + { + ProcessChPm25(rawData.ch_pm25); + } - if (rawData.ch_soil != null) - { - ProcessSoilMoisture(rawData.ch_soil); - } + if (rawData.ch_leak != null) + { + ProcessLeak(rawData.ch_leak); + } - if (rawData.ch_leaf != null) - { - ProcessLeafWet(rawData.ch_leaf); - } + if (rawData.ch_aisle != null) + { + ProcessExtraTempHum(rawData.ch_aisle, dataLastRead); + } - // Now do the stuff that requires more than one input parameter + if (rawData.ch_temp != null) + { + ProcessUserTemp(rawData.ch_temp); + } - // Only set the lightning time/distance if it is newer than what we already have - the GW1000 seems to reset this value - if (newLightningTime > LightningTime) - { - LightningTime = newLightningTime; - if (newLightningDistance < 999) - LightningDistance = newLightningDistance; - } + if (rawData.ch_soil != null) + { + ProcessSoilMoisture(rawData.ch_soil); + } - // Process outdoor temperature here, as GW1000 currently does not supply Dew Point so we have to calculate it in DoOutdoorTemp() - if (outdoortemp > -999) - DoOutdoorTemp(ConvertUnits.TempCToUser(outdoortemp), dataLastRead); + if (rawData.ch_leaf != null) + { + ProcessLeafWet(rawData.ch_leaf); + } - // Same for extra T/H sensors - for (var i = 1; i <= 8; i++) - { - if (ExtraHum[i] > 0) + // Now do the stuff that requires more than one input parameter + + // Only set the lightning time/distance if it is newer than what we already have - the GW1000 seems to reset this value + if (newLightningTime > LightningTime) { - var dp = MeteoLib.DewPoint(ConvertUnits.UserTempToC(ExtraTemp[i]), ExtraHum[i]); - ExtraDewPoint[i] = ConvertUnits.TempCToUser(dp); + LightningTime = newLightningTime; + if (newLightningDistance < 999) + LightningDistance = newLightningDistance; } - } - if (gustLast > -999 && windSpeedLast > -999 && windDirLast > -999) - { - DoWind(gustLast, windDirLast, windSpeedLast, dataLastRead); - } + // Process outdoor temperature here, as GW1000 currently does not supply Dew Point so we have to calculate it in DoOutdoorTemp() + if (outdoortemp > -999) + DoOutdoorTemp(ConvertUnits.TempCToUser(outdoortemp), dataLastRead); - if (rainLast > -999 && rainRateLast > -999) - { - DoRain(rainLast, rainRateLast, dataLastRead); - } + // Same for extra T/H sensors + for (var i = 1; i <= 8; i++) + { + if (ExtraHum[i] > 0) + { + var dp = MeteoLib.DewPoint(ConvertUnits.UserTempToC(ExtraTemp[i]), ExtraHum[i]); + ExtraDewPoint[i] = ConvertUnits.TempCToUser(dp); + } + } - if (outdoortemp > -999) - { - DoWindChill(windchill, dataLastRead); - DoApparentTemp(dataLastRead); - DoFeelsLike(dataLastRead); - DoHumidex(dataLastRead); - DoCloudBaseHeatIndex(dataLastRead); + if (gustLast > -999 && windSpeedLast > -999 && windDirLast > -999) + { + DoWind(gustLast, windDirLast, windSpeedLast, dataLastRead); + } - if (cumulus.StationOptions.CalculateSLP) + if (rainLast > -999 && rainRateLast > -999) { - var abs = cumulus.Calib.Press.Calibrate(StationPressure); - var slp = MeteoLib.GetSeaLevelPressure(AltitudeM(cumulus.Altitude), ConvertUnits.UserPressToMB(abs), ConvertUnits.UserTempToC(OutdoorTemperature), cumulus.Latitude); - DoPressure(ConvertUnits.PressMBToUser(slp), dataLastRead); + DoRain(rainLast, rainRateLast, dataLastRead); } - } - DoForecast("", false); + if (outdoortemp > -999) + { + DoWindChill(windchill, dataLastRead); + DoApparentTemp(dataLastRead); + DoFeelsLike(dataLastRead); + DoHumidex(dataLastRead); + DoCloudBaseHeatIndex(dataLastRead); - cumulus.BatteryLowAlarm.Triggered = batteryLow; + if (cumulus.StationOptions.CalculateSLP) + { + var abs = cumulus.Calib.Press.Calibrate(StationPressure); + var slp = MeteoLib.GetSeaLevelPressure(AltitudeM(cumulus.Altitude), ConvertUnits.UserPressToMB(abs), ConvertUnits.UserTempToC(OutdoorTemperature), cumulus.Latitude); + DoPressure(ConvertUnits.PressMBToUser(slp), dataLastRead); + } + } - UpdateStatusPanel(dataLastRead); - UpdateMQTT(); + DoForecast("", false); - dataReceived = true; - DataStopped = false; - cumulus.DataStoppedAlarm.Triggered = false; + cumulus.BatteryLowAlarm.Triggered = batteryLow; - var minute = DateTime.Now.Minute; - if (minute != lastMinute) - { - lastMinute = minute; + UpdateStatusPanel(dataLastRead); + UpdateMQTT(); - // at the start of every 20 minutes to trigger battery status check - if ((minute % 20) == 0 && !cumulus.cancellationToken.IsCancellationRequested) - { - _ = GetSensorIds(); - } + dataReceived = true; + DataStopped = false; + cumulus.DataStoppedAlarm.Triggered = false; - // every day dump the clock drift at midday each day - if (minute == 0 && DateTime.Now.Hour == 12) + var minute = DateTime.Now.Minute; + if (minute != lastMinute) { - GetSystemInfo(true); - } + lastMinute = minute; - var hour = DateTime.Now.Hour; - if (lastHour != hour) - { - lastHour = hour; + // at the start of every 20 minutes to trigger battery status check + if ((minute % 20) == 0 && !cumulus.cancellationToken.IsCancellationRequested) + { + _ = GetSensorIds(); + } - if (hour == 13) + // every day dump the clock drift at midday each day + if (minute == 0 && DateTime.Now.Hour == 12) { - var fw = GetFirmwareVersion(); - if (fw != "???") - { - GW1000FirmwareVersion = fw; - deviceModel = GW1000FirmwareVersion.Split('_')[0]; - deviceFirmware = GW1000FirmwareVersion.Split('_')[1]; + GetSystemInfo(true); + } - var fwString = GW1000FirmwareVersion.Split(underscoreV, StringSplitOptions.None); - if (fwString.Length > 1) - { - fwVersion = new Version(fwString[1]); - } - else + var hour = DateTime.Now.Hour; + if (lastHour != hour) + { + lastHour = hour; + + if (hour == 13) + { + var fw = GetFirmwareVersion(); + if (fw != "???") { - // failed to get the version, lets assume it's fairly new - fwVersion = new Version("1.6.5"); + GW1000FirmwareVersion = fw; + deviceModel = GW1000FirmwareVersion.Split('_')[0]; + deviceFirmware = GW1000FirmwareVersion.Split('_')[1]; + + var fwString = GW1000FirmwareVersion.Split(underscoreV, StringSplitOptions.None); + if (fwString.Length > 1) + { + fwVersion = new Version(fwString[1]); + } + else + { + // failed to get the version, lets assume it's fairly new + fwVersion = new Version("1.6.5"); + } } - } - _ = CheckAvailableFirmware(); + _ = CheckAvailableFirmware(); + } } } } @@ -510,7 +513,7 @@ private async Task GetSensorIds() LowBatteryDevices.Clear(); - if (sensors.Length > 0) + if (sensors != null && sensors.Length > 0) { for (var i = 0; i< sensors.Length; i++) { diff --git a/CumulusMX/EcowittLocalApi.cs b/CumulusMX/EcowittLocalApi.cs index b997c7e0..be95a9a7 100644 --- a/CumulusMX/EcowittLocalApi.cs +++ b/CumulusMX/EcowittLocalApi.cs @@ -145,7 +145,7 @@ public LiveData GetLiveData(CancellationToken token) string responseBody; int responseCode; - int retries = 3; + int retries = 2; int retry = 1; @@ -198,6 +198,10 @@ public LiveData GetLiveData(CancellationToken token) { cumulus.LogErrorMessage("GetLiveData: Error - This Station does not support the HTTP API!"); } + else + { + cumulus.LogExceptionMessage(ex, "GetLiveData: HTTP Error"); + } } catch (Exception ex) { @@ -263,6 +267,10 @@ public async Task GetSensorInfo(CancellationToken token) { cumulus.LogErrorMessage("GetSensorInfo: Error - This Station does not support the HTTP API!"); } + else + { + cumulus.LogExceptionMessage(ex, "GetSensorInfo: HTTP Error"); + } } catch (Exception ex) { diff --git a/Updates.txt b/Updates.txt index 69c87fda..8ba3ddf2 100644 --- a/Updates.txt +++ b/Updates.txt @@ -1,7 +1,12 @@ -4.2.0 - b4033 +4.2.0 - b4034 ————————————— -BETA b4033 Changes: -——————————————————— +BETA 4034 Changes: +—————————————————— +- Improved Ecowitt HTTP API station error handling/recovery when station goes offline + + +BETA 4033 Changes: +—————————————————— - Add Ecowitt HTTP API password - Add polling time update to Ecowitt HTTP API station - Add more Ecowitt HTTP API station error handling @@ -45,7 +50,7 @@ Fixed Package Updates - SQLite: Reverted to v2.1.8 pending fix from author - +- MQTTnet 4.1.3 - b4028 From 510b605bcba3234ecbc0e8f279667068341bdfc0 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Wed, 11 Sep 2024 19:06:07 +0100 Subject: [PATCH 32/63] New web tag #MonthRainfall Fix Ecowitt HTTP API station parsing of some decimal values when running in a user locale that uses comma decimals --- CumulusMX/CumulusMX.csproj | 2 +- CumulusMX/EcowittApi.cs | 6 +++ CumulusMX/EcowittLocalApi.cs | 7 +-- CumulusMX/webtags.cs | 87 ++++++++++++++++++++++++++++-------- Updates.txt | 16 ++++++- 5 files changed, 95 insertions(+), 23 deletions(-) diff --git a/CumulusMX/CumulusMX.csproj b/CumulusMX/CumulusMX.csproj index fea08e75..77e50499 100644 --- a/CumulusMX/CumulusMX.csproj +++ b/CumulusMX/CumulusMX.csproj @@ -55,7 +55,7 @@ - 4.2.0.4034-beta + 4.2.0.4035-beta Copyright © 2015-$([System.DateTime]::Now.ToString('yyyy')) Cumulus MX $(AssemblyName) diff --git a/CumulusMX/EcowittApi.cs b/CumulusMX/EcowittApi.cs index a6e86c40..07fc8e60 100644 --- a/CumulusMX/EcowittApi.cs +++ b/CumulusMX/EcowittApi.cs @@ -2505,6 +2505,12 @@ internal string[] GetStationList(bool CheckCamera, string macAddress, Cancellati vers = stn.stationtype.Split('V')[^1]; model = stn.stationtype.Replace("_", string.Empty).Split('V')[0]; } + else + { + // no idea what we got! + vers = stn.name; + model = stn.name; + } } cumulus.LogDebugMessage($"API.GetStationList: Found vers={vers}, model={model}"); diff --git a/CumulusMX/EcowittLocalApi.cs b/CumulusMX/EcowittLocalApi.cs index be95a9a7..505bbc1d 100644 --- a/CumulusMX/EcowittLocalApi.cs +++ b/CumulusMX/EcowittLocalApi.cs @@ -1,4 +1,5 @@ using System; +using System.Globalization; using System.Threading; using System.Threading.Tasks; @@ -10,7 +11,7 @@ namespace CumulusMX internal sealed class EcowittLocalApi : IDisposable { private readonly Cumulus cumulus; - + private static NumberFormatInfo invNum = CultureInfo.InvariantCulture.NumberFormat; public EcowittLocalApi(Cumulus cumul) { cumulus = cumul; @@ -521,7 +522,7 @@ public double? valDbl unit = temp[1]; val = temp[0]; } - return double.TryParse(val, out double result) ? result : null; + return double.TryParse(val, invNum, out double result) ? result : null; } } } @@ -574,7 +575,7 @@ public double? distanceVal get { var temp = distance.Split(' '); - return double.TryParse(temp[0], out double result) ? result : null; + return double.TryParse(temp[0], invNum, out double result) ? result : null; } } diff --git a/CumulusMX/webtags.cs b/CumulusMX/webtags.cs index 1233a8cd..8bfdb3de 100644 --- a/CumulusMX/webtags.cs +++ b/CumulusMX/webtags.cs @@ -3595,28 +3595,38 @@ private string TagMonthTempAvg(Dictionary tagParams) var month = tagParams.Get("m"); DateTime start; DateTime end; + double avg; - if (year != null && month != null) - { - start = new DateTime(int.Parse(year), int.Parse(month), 1, 0, 0, 0, DateTimeKind.Local); - end = start.AddMonths(1); - } - else + try { - end = DateTime.Now.Date; - start = new DateTime(end.Year, end.Month, 1, 0, 0, 0, DateTimeKind.Local); - } + if (year != null && month != null) + { + var yr = int.Parse(year); + var mon = int.Parse(month); - if (start.Date == DateTime.Now.AddHours(cumulus.GetHourInc()).Date) - { - // first day of the current month, there are no dayfile entries - // so return the average temp so far today - return Tagavgtemp(tagParams); - } + if (yr > 1970 && yr <= DateTime.Now.Year && mon > 0 && mon < 13) + { + start = new DateTime(int.Parse(year), int.Parse(month), 1, 0, 0, 0, DateTimeKind.Local); + end = start.AddMonths(1); + } + else + { + return "-"; + } + } + else + { + end = DateTime.Now.Date; + start = new DateTime(end.Year, end.Month, 1, 0, 0, 0, DateTimeKind.Local); + } + + if (start.Date == DateTime.Now.AddHours(cumulus.GetHourInc()).Date) + { + // first day of the current month, there are no dayfile entries + // so return the average temp so far today + return Tagavgtemp(tagParams); + } - double avg; - try - { avg = station.DayFile.Where(x => x.Date >= start && x.Date < end).Average(rec => rec.AvgTemp); } catch @@ -3649,6 +3659,46 @@ private string TagYearTempAvg(Dictionary tagParams) return CheckRcDp(CheckTempUnit(avg, tagParams), tagParams, cumulus.TempDPlaces); } + private string TagMonthRainfall(Dictionary tagParams) + { + var year = tagParams.Get("y"); + var month = tagParams.Get("m"); + DateTime start; + DateTime end; + double total; + + if (year != null && month != null) + { + try + { + var yr = int.Parse(year); + var mon = int.Parse(month); + + if (yr > 1970 && yr <= DateTime.Now.Year && mon > 0 && mon < 13) + { + start = new DateTime(yr, mon, 1, 0, 0, 0, DateTimeKind.Local); + end = start.AddMonths(1); + total = station.DayFile.Where(x => x.Date >= start && x.Date < end).Sum(rec => rec.TotalRain); + } + else + { + return "-"; + } + } + catch + { + // error or no data found + return "-"; + } + } + else + { + total = station.RainMonth; + } + + return CheckRcDp(CheckTempUnit(total, tagParams), tagParams, cumulus.RainDPlaces); + } + private string TagAnnualRainfall(Dictionary tagParams) { var year = tagParams.Get("y"); @@ -6793,6 +6843,7 @@ public void InitialiseWebtags() // Specifc Month/Year values { "MonthTempAvg", TagMonthTempAvg }, { "YearTempAvg", TagYearTempAvg }, + { "MonthRainfall", TagMonthRainfall }, { "AnnualRainfall", TagAnnualRainfall }, // Options { "Option_useApparent", TagOption_useApparent }, diff --git a/Updates.txt b/Updates.txt index 8ba3ddf2..05853866 100644 --- a/Updates.txt +++ b/Updates.txt @@ -1,8 +1,18 @@ -4.2.0 - b4034 +4.2.0 - b4035 ————————————— +BETA 4034 Changes: +—————————————————— +- Fixes to the Interval/Daily data viewer web pages +- Fix Ecowitt HTTP API station parsing of some decimal values when running in a user locale that uses comma decimals + + + BETA 4034 Changes: —————————————————— - Improved Ecowitt HTTP API station error handling/recovery when station goes offline +- Daily and Interval data viewers now visible on the dashboard +- Removed Ecowitt HTTP API station password +- New web tag <#MonthRainfall> BETA 4033 Changes: @@ -33,6 +43,10 @@ New - The format is a comma separated list of sensor/transmitter IDs and battery states - Eg. "wh80 - ,wh41ch1-,wh41ch2-" - Where is "LOW" or "0" or "1" depending on what the sensor sends +- New web tag <#MonthRainfall> which returns the rainfall total for the current month by default + - Takes optional parameters y=YYYY and m=MM (both must be specified) to return the total rainfall for specified month in the specified year + - Eg. <#MonthRainfall y=2018 m=10> + Changed - Davis WLL/Davis Cloud stations now use the WL.com subscription level to determine if they use WL.com as a logger From e6edeef829b4359d807a1f883b33930661c35dfd Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Wed, 11 Sep 2024 19:40:36 +0100 Subject: [PATCH 33/63] Fix Ecowitt HTTP API station settings not having a field for the MAC address --- CumulusMX/StationSettings.cs | 3 +++ Updates.txt | 2 +- 2 files changed, 4 insertions(+), 1 deletion(-) diff --git a/CumulusMX/StationSettings.cs b/CumulusMX/StationSettings.cs index 25c7ec73..4b849470 100644 --- a/CumulusMX/StationSettings.cs +++ b/CumulusMX/StationSettings.cs @@ -160,6 +160,7 @@ internal string GetAlpacaFormData() var ecowittHttpApi = new JsonStationSettingsHttpApi() { ipaddress = cumulus.Gw1000IpAddress, + macaddress = cumulus.Gw1000MacAddress, password = cumulus.EcowittHttpPassword }; @@ -918,6 +919,7 @@ internal string UpdateConfig(IHttpContext context) if (settings.ecowitthttpapi != null) { cumulus.Gw1000IpAddress = string.IsNullOrWhiteSpace(settings.ecowitthttpapi.ipaddress) ? null : settings.ecowitthttpapi.ipaddress.Trim(); + cumulus.Gw1000MacAddress = string.IsNullOrWhiteSpace(settings.ecowitthttpapi.macaddress) ? null : settings.ecowitthttpapi.macaddress.Trim().ToUpper(); cumulus.EcowittHttpPassword = string.IsNullOrWhiteSpace(settings.ecowitthttpapi.password) ? null : settings.ecowitthttpapi.password.Trim(); } } @@ -1807,6 +1809,7 @@ internal class JsonStationSettingsGw1000Conn internal class JsonStationSettingsHttpApi { public string ipaddress { get; set; } + public string macaddress { get; set; } public string password { get; set; } } diff --git a/Updates.txt b/Updates.txt index 05853866..4eede72f 100644 --- a/Updates.txt +++ b/Updates.txt @@ -4,7 +4,7 @@ BETA 4034 Changes: —————————————————— - Fixes to the Interval/Daily data viewer web pages - Fix Ecowitt HTTP API station parsing of some decimal values when running in a user locale that uses comma decimals - +- Fix Ecowitt HTTP API station settings not having a field for the MAC address BETA 4034 Changes: From 4d55c5c9279a9afd7da679a42438a08b2a7e1ccb Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Thu, 12 Sep 2024 16:33:16 +0100 Subject: [PATCH 34/63] Add support for Ecowitt srain_piezo to Ecowitt HTTP API, and HTTP Custom Sender stations --- CumulusMX/Cumulus.cs | 4 +++- CumulusMX/EcowittHttpApiStation.cs | 7 +++++++ CumulusMX/HttpStationEcowitt.cs | 7 +++++++ CumulusMX/StationSettings.cs | 8 +++++++- Updates.txt | 5 +++-- 5 files changed, 27 insertions(+), 4 deletions(-) diff --git a/CumulusMX/Cumulus.cs b/CumulusMX/Cumulus.cs index 764c4ef0..6188c099 100644 --- a/CumulusMX/Cumulus.cs +++ b/CumulusMX/Cumulus.cs @@ -3969,6 +3969,7 @@ private void ReadIniFile() Gw1000AutoUpdateIpAddress = ini.GetValue("GW1000", "AutoUpdateIpAddress", true); Gw1000PrimaryTHSensor = ini.GetValue("GW1000", "PrimaryTHSensor", 0, 0, 99); // 0=default, 1-8=extra t/h sensor number, 99=use indoor sensor Gw1000PrimaryRainSensor = ini.GetValue("GW1000", "PrimaryRainSensor", 0, 0, 1); //0=main station (tipping bucket) 1=piezo + EcowittIsRainingUsePiezo = ini.GetValue("GW1000", "UsePiezoIsRaining", false); EcowittExtraEnabled = ini.GetValue("GW1000", "ExtraSensorDataEnabled", false); EcowittCloudExtraEnabled = ini.GetValue("GW1000", "ExtraCloudSensorDataEnabled", false); EcowittExtraUseSolar = ini.GetValue("GW1000", "ExtraSensorUseSolar", true); @@ -5667,6 +5668,7 @@ internal void WriteIniFile() ini.SetValue("GW1000", "AutoUpdateIpAddress", Gw1000AutoUpdateIpAddress); ini.SetValue("GW1000", "PrimaryTHSensor", Gw1000PrimaryTHSensor); ini.SetValue("GW1000", "PrimaryRainSensor", Gw1000PrimaryRainSensor); + ini.SetValue("GW1000", "UsePiezoIsRaining", EcowittIsRainingUsePiezo); ini.SetValue("GW1000", "ExtraSensorDataEnabled", EcowittExtraEnabled); ini.SetValue("GW1000", "ExtraCloudSensorDataEnabled", EcowittCloudExtraEnabled); ini.SetValue("GW1000", "ExtraSensorUseSolar", EcowittExtraUseSolar); @@ -7180,8 +7182,8 @@ public void WriteStringsFile() public bool EcowittExtraUseMainForwarders { get; set; } public string[] EcowittExtraForwarders { get; set; } = new string[10]; public string EcowittHttpPassword { get; set; } - public int[] EcowittMapWN34 { get; set; } = new int[9]; + public bool EcowittIsRainingUsePiezo { get; set; } public bool AmbientExtraEnabled { get; set; } public bool AmbientExtraUseSolar { get; set; } diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index 450a4796..e5b87225 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -989,6 +989,13 @@ private void ProcessRain(EcowittLocalApi.CommonSensor[] sensors) } break; + case "srain_piezo": + if (cumulus.EcowittIsRainingUsePiezo) + { + IsRaining = sensor.val == "1"; + } + break; + case "0x10": // Rain day case "0x11": // Rain week case "0x12": // Rain month diff --git a/CumulusMX/HttpStationEcowitt.cs b/CumulusMX/HttpStationEcowitt.cs index b3899ca7..598f423b 100644 --- a/CumulusMX/HttpStationEcowitt.cs +++ b/CumulusMX/HttpStationEcowitt.cs @@ -618,6 +618,8 @@ public string ApplyData(string dataString, bool main, DateTime? ts = null) // wrain_piezo // mrain_piezo // yrain_piezo + // plus + // srain_piezo string rain, rRate; // if no yearly counter, try the total counter @@ -659,6 +661,11 @@ public string ApplyData(string dataString, bool main, DateTime? ts = null) var rateVal = ConvertUnits.RainINToUser(Convert.ToDouble(rRate, invNum)); DoRain(rainVal, rateVal, recDate); } + + if (data["srain_piezo"] != null && cumulus.EcowittIsRainingUsePiezo) + { + IsRaining = data["srain_piezo"] == "1"; + } } catch (Exception ex) { diff --git a/CumulusMX/StationSettings.cs b/CumulusMX/StationSettings.cs index 4b849470..79e29da8 100644 --- a/CumulusMX/StationSettings.cs +++ b/CumulusMX/StationSettings.cs @@ -161,7 +161,8 @@ internal string GetAlpacaFormData() { ipaddress = cumulus.Gw1000IpAddress, macaddress = cumulus.Gw1000MacAddress, - password = cumulus.EcowittHttpPassword + password = cumulus.EcowittHttpPassword, + piezosrain = cumulus.EcowittIsRainingUsePiezo }; var ecowitt = new JsonStationSettingsEcowitt @@ -170,6 +171,7 @@ internal string GetAlpacaFormData() gwaddr = cumulus.EcowittGatewayAddr, localaddr = cumulus.EcowittLocalAddr, interval = cumulus.EcowittCustomInterval, + piezosrain = cumulus.EcowittIsRainingUsePiezo, forward = [] }; @@ -921,6 +923,7 @@ internal string UpdateConfig(IHttpContext context) cumulus.Gw1000IpAddress = string.IsNullOrWhiteSpace(settings.ecowitthttpapi.ipaddress) ? null : settings.ecowitthttpapi.ipaddress.Trim(); cumulus.Gw1000MacAddress = string.IsNullOrWhiteSpace(settings.ecowitthttpapi.macaddress) ? null : settings.ecowitthttpapi.macaddress.Trim().ToUpper(); cumulus.EcowittHttpPassword = string.IsNullOrWhiteSpace(settings.ecowitthttpapi.password) ? null : settings.ecowitthttpapi.password.Trim(); + cumulus.EcowittIsRainingUsePiezo = settings.ecowitthttpapi.piezosrain; } } catch (Exception ex) @@ -940,6 +943,7 @@ internal string UpdateConfig(IHttpContext context) cumulus.EcowittGatewayAddr = string.IsNullOrWhiteSpace(settings.ecowitt.gwaddr) ? null : settings.ecowitt.gwaddr.Trim(); cumulus.EcowittLocalAddr = string.IsNullOrWhiteSpace(settings.ecowitt.localaddr) ? null : settings.ecowitt.localaddr.Trim(); cumulus.EcowittCustomInterval = settings.ecowitt.interval; + cumulus.EcowittIsRainingUsePiezo = settings.ecowitthttpapi.piezosrain; for (var i = 0; i < 10; i++) { @@ -1811,6 +1815,7 @@ internal class JsonStationSettingsHttpApi public string ipaddress { get; set; } public string macaddress { get; set; } public string password { get; set; } + public bool piezosrain { get; set; } } internal class JsonStationSettingsEcowitt @@ -1819,6 +1824,7 @@ internal class JsonStationSettingsEcowitt public string gwaddr { get; set; } public string localaddr { get; set; } public int interval { get; set; } + public bool piezosrain { get; set; } public List forward { get; set; } } diff --git a/Updates.txt b/Updates.txt index 4eede72f..dced2eec 100644 --- a/Updates.txt +++ b/Updates.txt @@ -1,10 +1,11 @@ 4.2.0 - b4035 ————————————— -BETA 4034 Changes: +BETA 4035 Changes: —————————————————— - Fixes to the Interval/Daily data viewer web pages - Fix Ecowitt HTTP API station parsing of some decimal values when running in a user locale that uses comma decimals - Fix Ecowitt HTTP API station settings not having a field for the MAC address +- Add support for Ecowitt srain_piezo to Ecowitt HTTP API, and HTTP Custom Sender stations BETA 4034 Changes: @@ -46,7 +47,7 @@ New - New web tag <#MonthRainfall> which returns the rainfall total for the current month by default - Takes optional parameters y=YYYY and m=MM (both must be specified) to return the total rainfall for specified month in the specified year - Eg. <#MonthRainfall y=2018 m=10> - +- Support for Ecowitt WS90 piezo IsRaining status to trigger MX IsRaining. Currently only supported with a WS90 connected to a GW2000 Changed - Davis WLL/Davis Cloud stations now use the WL.com subscription level to determine if they use WL.com as a logger From 57a36f056036c5657d5b7015f746377e18b6fb99 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Thu, 12 Sep 2024 22:26:21 +0100 Subject: [PATCH 35/63] Move piezo_srain to the sensor mappings section of settings --- CumulusMX/GW1000Station.cs | 17 +++++++++-------- CumulusMX/StationSettings.cs | 11 ++++------- 2 files changed, 13 insertions(+), 15 deletions(-) diff --git a/CumulusMX/GW1000Station.cs b/CumulusMX/GW1000Station.cs index 212dd2a4..995a63b5 100644 --- a/CumulusMX/GW1000Station.cs +++ b/CumulusMX/GW1000Station.cs @@ -673,6 +673,8 @@ private void GetSensorIdsNew() // ... etc // (??) - 0x?? - checksum + //data = [0xFF, 0xFF, 0x3C, 0x01, 0x62, 0x00, 0x00, 0x00, 0x00, 0xF0, 0xFF, 0x00, 0x01, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x02, 0x00, 0x0C, 0x00, 0x2E, 0xA4, 0x04, 0x03, 0x00, 0x00, 0xC6, 0xA4, 0x8D, 0x04, 0x04, 0x00, 0x00, 0x00, 0x8B, 0x00, 0x04, 0x05, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x06, 0x00, 0x00, 0x00, 0x90, 0x00, 0x04, 0x07, 0x00, 0x00, 0x00, 0xEE, 0x00, 0x04, 0x08, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x09, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x0A, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x0B, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x0C, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x0D, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x0E, 0x00, 0x44, 0xA9, 0x12, 0x1F, 0x00, 0x0F, 0xFF, 0xFF, 0xFF, 0xFF, 0x1F, 0x00, 0x10, 0xFF, 0xFF, 0xFF, 0xFF, 0x1F, 0x00, 0x11, 0xFF, 0xFF, 0xFF, 0xFF, 0x1F, 0x00, 0x12, 0xFF, 0xFF, 0xFF, 0xFF, 0x1F, 0x00, 0x13, 0xFF, 0xFF, 0xFF, 0xFF, 0x1F, 0x00, 0x14, 0xFF, 0xFF, 0xFF, 0xFF, 0x1F, 0x00, 0x15, 0xFF, 0xFF, 0xFF, 0xFF, 0x1F, 0x00, 0x16, 0x00, 0x0A, 0xB2, 0x12, 0x0F, 0x00, 0x17, 0x00, 0x00, 0x00, 0x8A, 0x0F, 0x00, 0x18, 0xFF, 0xFF, 0xFF, 0xFF, 0x0F, 0x00, 0x19, 0xFF, 0xFF, 0xFF, 0xFF, 0x0F, 0x00, 0x1A, 0x00, 0x00, 0xEB, 0xBC, 0x04, 0x03, 0x1B, 0xFF, 0xFF, 0xFF, 0xFF, 0x0F, 0x00, 0x1C, 0xFF, 0xFF, 0xFF, 0xFF, 0x0F, 0x00, 0x1D, 0xFF, 0xFF, 0xFF, 0xFF, 0x0F, 0x00, 0x1E, 0xFF, 0xFF, 0xFF, 0xFF, 0x0F, 0x00, 0x1F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x20, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x21, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x22, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x23, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x24, 0x00, 0x9D, 0xAD, 0x9A, 0xFF, 0x00, 0x25, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x26, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x27, 0xFF, 0xFF, 0xFF, 0xFF, 0x0F, 0x00, 0x28, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x29, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x2A, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x2B, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x2C, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x2D, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x2E, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x2F, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x30, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x31, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x02]; + var batteryLow = false; try @@ -733,14 +735,11 @@ private bool PrintSensorInfoNew(byte[] data, int idx) } catch { - // leave the device id as null + // uknown sensor + type = $"[unknown sensor = {data[idx]}]"; } var battPos = idx + 5; var sigPos = idx + 6; - if (string.IsNullOrEmpty(type)) - { - type = $"[unknown sensor = {data[idx]}]"; - } // Wh65 could be a Wh65 or a Wh24, we found out using the System Info command if (type == "WH65") { @@ -787,6 +786,7 @@ private bool PrintSensorInfoNew(byte[] data, int idx) case "WH65": case "WH24": case "WH26": + case "WH24/WH65": batt = TestBattery1(data[battPos], 1); // 0 or 1 break; @@ -842,6 +842,7 @@ private bool PrintSensorInfoNew(byte[] data, int idx) batt = $"{data[battPos]} ({TestBattery3(data[battPos])})"; // 0-5 (6+9), low = 1 break; default: + // Don't know what this is batt = "???"; break; } @@ -868,9 +869,9 @@ private void GetLiveData() byte[] data = Api.DoCommand(GW1000Api.Commands.CMD_GW1000_LIVEDATA); // sample data = in-temp, in-hum, abs-baro, rel-baro, temp, hum, dir, speed, gust, light, UV uW, UV-I, rain-rate, rain-day, rain-week, rain-month, rain-year, PM2.5, PM-ch1, Soil-1, temp-2, hum-2, temp-3, hum-3, batt - //byte[] data = new byte[] { 0xFF,0xFF,0x27,0x00,0x5D,0x01,0x00,0x83,0x06,0x55,0x08,0x26,0xE7,0x09,0x26,0xDC,0x02,0x00,0x5D,0x07,0x61,0x0A,0x00,0x89,0x0B,0x00,0x19,0x0C,0x00,0x25,0x15,0x00,0x00,0x00,0x00,0x16,0x00,0x00,0x17,0x00,0x0E,0x00,0x3C,0x10,0x00,0x1E,0x11,0x01,0x4A,0x12,0x00,0x00,0x02,0x68,0x13,0x00,0x00,0x14,0xDC,0x2A,0x01,0x90,0x4D,0x00,0xE3,0x2C,0x34,0x1B,0x00,0xD3,0x23,0x3C,0x1C,0x00,0x60,0x24,0x5A,0x4C,0x04,0x00,0x00,0x00,0xFF,0x5C,0xFF,0x00,0xF4,0xFF,0xFF,0xFF,0xFF,0xFF,0x00,0x00,0xBA } - //byte[] data = new byte[] { 0xFF, 0xFF, 0x27, 0x00, 0x6D, 0x01, 0x00, 0x96, 0x06, 0x3C, 0x08, 0x27, 0x00, 0x09, 0x27, 0x49, 0x02, 0x00, 0x16, 0x07, 0x61, 0x0A, 0x00, 0x62, 0x0B, 0x00, 0x00, 0x0C, 0x00, 0x06, 0x15, 0x00, 0x01, 0x7D, 0x40, 0x16, 0x00, 0x00, 0x17, 0x00, 0x0E, 0x00, 0x00, 0x10, 0x00, 0x00, 0x11, 0x00, 0xF7, 0x12, 0x00, 0x00, 0x01, 0x5C, 0x13, 0x00, 0x00, 0x15, 0x54, 0x2A, 0x06, 0x40, 0x4D, 0x00, 0xAB, 0x1A, 0xFF, 0x3E, 0x22, 0x39, 0x1B, 0x00, 0x3D, 0x23, 0x51, 0x1C, 0x00, 0xA0, 0x24, 0x45, 0x1D, 0x00, 0xA4, 0x25, 0x3C, 0x1E, 0x00, 0x9D, 0x26, 0x3E, 0x4C, 0x04, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xA4, 0x00, 0xF4, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x19, 0x00, 0x1A, 0x8F } - //byte[] data = new byte[] { 0xFF, 0xFF, 0x27, 0x00, 0x6D, 0x01, 0x00, 0xF8, 0x06, 0x35, 0x08, 0x27, 0xD6, 0x09, 0x27, 0xE1, 0x02, 0x00, 0xD2, 0x07, 0x5E, 0x0A, 0x00, 0x79, 0x0B, 0x00, 0x05, 0x0C, 0x00, 0x05, 0x15, 0x00, 0x00, 0x00, 0x00, 0x16, 0x00, 0x02, 0x17, 0x00, 0x2A, 0x01, 0x71, 0x4D, 0x00, 0xC4, 0x1A, 0x00, 0xE4, 0x22, 0x3B, 0x4C, 0x05, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0x06, 0xF5, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x19, 0x00, 0x33, 0x0E, 0x00, 0x00, 0x10, 0x00, 0x00, 0x11, 0x00, 0x0D, 0x12, 0x00, 0x00, 0x01, 0x0B, 0x13, 0x00, 0x00, 0x3A, 0x62, 0x0D, 0x00, 0x00, 0x70, 0x00, 0xED, 0x3A, 0x00, 0x2B, 0x00, 0x11, 0x00, 0x1E, 0x00, 0x0D, 0x03, 0x7B, 0x03, 0xD2, 0x06, 0x02 } + //byte[] data = [0xFF, 0xFF, 0x27, 0x00, 0x6D, 0x01, 0x00, 0x96, 0x06, 0x3C, 0x08, 0x27, 0x00, 0x09, 0x27, 0x49, 0x02, 0x00, 0x16, 0x07, 0x61, 0x0A, 0x00, 0x62, 0x0B, 0x00, 0x00, 0x0C, 0x00, 0x06, 0x15, 0x00, 0x01, 0x7D, 0x40, 0x16, 0x00, 0x00, 0x17, 0x00, 0x0E, 0x00, 0x00, 0x10, 0x00, 0x00, 0x11, 0x00, 0xF7, 0x12, 0x00, 0x00, 0x01, 0x5C, 0x13, 0x00, 0x00, 0x15, 0x54, 0x2A, 0x06, 0x40, 0x4D, 0x00, 0xAB, 0x1A, 0xFF, 0x3E, 0x22, 0x39, 0x1B, 0x00, 0x3D, 0x23, 0x51, 0x1C, 0x00, 0xA0, 0x24, 0x45, 0x1D, 0x00, 0xA4, 0x25, 0x3C, 0x1E, 0x00, 0x9D, 0x26, 0x3E, 0x4C, 0x04, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xA4, 0x00, 0xF4, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x19, 0x00, 0x1A, 0x8F ] + //byte[] data = [0xFF,0xFF,0x27,0x00,0x5D,0x01,0x00,0x83,0x06,0x55,0x08,0x26,0xE7,0x09,0x26,0xDC,0x02,0x00,0x5D,0x07,0x61,0x0A,0x00,0x89,0x0B,0x00,0x19,0x0C,0x00,0x25,0x15,0x00,0x00,0x00,0x00,0x16,0x00,0x00,0x17,0x00,0x0E,0x00,0x3C,0x10,0x00,0x1E,0x11,0x01,0x4A,0x12,0x00,0x00,0x02,0x68,0x13,0x00,0x00,0x14,0xDC,0x2A,0x01,0x90,0x4D,0x00,0xE3,0x2C,0x34,0x1B,0x00,0xD3,0x23,0x3C,0x1C,0x00,0x60,0x24,0x5A,0x4C,0x04,0x00,0x00,0x00,0xFF,0x5C,0xFF,0x00,0xF4,0xFF,0xFF,0xFF,0xFF,0xFF,0x00,0x00,0xBA ] + //byte[] data = [0xFF, 0xFF, 0x27, 0x00, 0x6D, 0x01, 0x00, 0xF8, 0x06, 0x35, 0x08, 0x27, 0xD6, 0x09, 0x27, 0xE1, 0x02, 0x00, 0xD2, 0x07, 0x5E, 0x0A, 0x00, 0x79, 0x0B, 0x00, 0x05, 0x0C, 0x00, 0x05, 0x15, 0x00, 0x00, 0x00, 0x00, 0x16, 0x00, 0x02, 0x17, 0x00, 0x2A, 0x01, 0x71, 0x4D, 0x00, 0xC4, 0x1A, 0x00, 0xE4, 0x22, 0x3B, 0x4C, 0x05, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0x06, 0xF5, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x19, 0x00, 0x33, 0x0E, 0x00, 0x00, 0x10, 0x00, 0x00, 0x11, 0x00, 0x0D, 0x12, 0x00, 0x00, 0x01, 0x0B, 0x13, 0x00, 0x00, 0x3A, 0x62, 0x0D, 0x00, 0x00, 0x70, 0x00, 0xED, 0x3A, 0x00, 0x2B, 0x00, 0x11, 0x00, 0x1E, 0x00, 0x0D, 0x03, 0x7B, 0x03, 0xD2, 0x06, 0x02 ] // expected response // 0 - 0xff - header diff --git a/CumulusMX/StationSettings.cs b/CumulusMX/StationSettings.cs index 79e29da8..59bbecd9 100644 --- a/CumulusMX/StationSettings.cs +++ b/CumulusMX/StationSettings.cs @@ -140,6 +140,7 @@ internal string GetAlpacaFormData() { primaryTHsensor = cumulus.Gw1000PrimaryTHSensor, primaryRainSensor = cumulus.Gw1000PrimaryRainSensor, + piezosrain = cumulus.EcowittIsRainingUsePiezo, wn34chan1 = cumulus.EcowittMapWN34[1], wn34chan2 = cumulus.EcowittMapWN34[2], wn34chan3 = cumulus.EcowittMapWN34[3], @@ -161,8 +162,7 @@ internal string GetAlpacaFormData() { ipaddress = cumulus.Gw1000IpAddress, macaddress = cumulus.Gw1000MacAddress, - password = cumulus.EcowittHttpPassword, - piezosrain = cumulus.EcowittIsRainingUsePiezo + password = cumulus.EcowittHttpPassword }; var ecowitt = new JsonStationSettingsEcowitt @@ -171,7 +171,6 @@ internal string GetAlpacaFormData() gwaddr = cumulus.EcowittGatewayAddr, localaddr = cumulus.EcowittLocalAddr, interval = cumulus.EcowittCustomInterval, - piezosrain = cumulus.EcowittIsRainingUsePiezo, forward = [] }; @@ -923,7 +922,6 @@ internal string UpdateConfig(IHttpContext context) cumulus.Gw1000IpAddress = string.IsNullOrWhiteSpace(settings.ecowitthttpapi.ipaddress) ? null : settings.ecowitthttpapi.ipaddress.Trim(); cumulus.Gw1000MacAddress = string.IsNullOrWhiteSpace(settings.ecowitthttpapi.macaddress) ? null : settings.ecowitthttpapi.macaddress.Trim().ToUpper(); cumulus.EcowittHttpPassword = string.IsNullOrWhiteSpace(settings.ecowitthttpapi.password) ? null : settings.ecowitthttpapi.password.Trim(); - cumulus.EcowittIsRainingUsePiezo = settings.ecowitthttpapi.piezosrain; } } catch (Exception ex) @@ -943,7 +941,6 @@ internal string UpdateConfig(IHttpContext context) cumulus.EcowittGatewayAddr = string.IsNullOrWhiteSpace(settings.ecowitt.gwaddr) ? null : settings.ecowitt.gwaddr.Trim(); cumulus.EcowittLocalAddr = string.IsNullOrWhiteSpace(settings.ecowitt.localaddr) ? null : settings.ecowitt.localaddr.Trim(); cumulus.EcowittCustomInterval = settings.ecowitt.interval; - cumulus.EcowittIsRainingUsePiezo = settings.ecowitthttpapi.piezosrain; for (var i = 0; i < 10; i++) { @@ -973,6 +970,7 @@ internal string UpdateConfig(IHttpContext context) { cumulus.Gw1000PrimaryTHSensor = settings.ecowittmaps.primaryTHsensor; cumulus.Gw1000PrimaryRainSensor = settings.ecowittmaps.primaryRainSensor; + cumulus.EcowittIsRainingUsePiezo = settings.ecowittmaps.piezosrain; if (cumulus.EcowittMapWN34[1] != settings.ecowittmaps.wn34chan1) { @@ -1815,7 +1813,6 @@ internal class JsonStationSettingsHttpApi public string ipaddress { get; set; } public string macaddress { get; set; } public string password { get; set; } - public bool piezosrain { get; set; } } internal class JsonStationSettingsEcowitt @@ -1824,7 +1821,6 @@ internal class JsonStationSettingsEcowitt public string gwaddr { get; set; } public string localaddr { get; set; } public int interval { get; set; } - public bool piezosrain { get; set; } public List forward { get; set; } } @@ -1844,6 +1840,7 @@ public class JsonStationSettingsEcowittMappings { public int primaryTHsensor { get; set; } public int primaryRainSensor { get; set; } + public bool piezosrain { get; set; } public int wn34chan1 { get; set; } public int wn34chan2 { get; set; } From 2d54970625c8696833da924baa6fcc0be259ec52 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Fri, 13 Sep 2024 10:09:11 +0100 Subject: [PATCH 36/63] HTTP API - more invariant double conversions --- CumulusMX/EcowittHttpApiStation.cs | 19 +++++++++++-------- CumulusMX/EcowittLocalApi.cs | 2 +- 2 files changed, 12 insertions(+), 9 deletions(-) diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index e5b87225..df818c19 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -1,4 +1,5 @@ using System; +using System.Globalization; using System.Linq; using System.Threading; using System.Threading.Tasks; @@ -11,6 +12,8 @@ namespace CumulusMX internal class EcowittHttpApiStation : WeatherStation #pragma warning restore CA1001 // Types that own disposable fields should be disposable { + private static readonly NumberFormatInfo invNum = CultureInfo.InvariantCulture.NumberFormat; + private string deviceModel; private string deviceFirmware; private int updateRate = 10000; // 10 seconds by default @@ -690,7 +693,7 @@ private void ProcessCommonList(EcowittLocalApi.CommonSensor[] sensors, DateTime break; case "0x0B": //Wind Speed (val unit) var arr = sensor.val.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out var valDbl)) + if (arr.Length == 2 && double.TryParse(arr[0], invNum, out var valDbl)) { var spd = arr[1] switch { @@ -709,7 +712,7 @@ private void ProcessCommonList(EcowittLocalApi.CommonSensor[] sensors, DateTime break; case "0x0C": // Gust speed (val unit) arr = sensor.val.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out valDbl)) + if (arr.Length == 2 && double.TryParse(arr[0], invNum, out valDbl)) { var spd = arr[1] switch { @@ -728,7 +731,7 @@ private void ProcessCommonList(EcowittLocalApi.CommonSensor[] sensors, DateTime break; case "0x15": //Light (value unit) arr = sensor.val.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out valDbl)) + if (arr.Length == 2 && double.TryParse(arr[0], invNum, out valDbl)) { var light = arr[1] switch { @@ -827,7 +830,7 @@ private void ProcessWh25(EcowittLocalApi.Wh25Sensor[] sensors, DateTime dateTime if (sensor.rel != null && !cumulus.StationOptions.CalculateSLP) { var arr = sensor.rel.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out var val)) + if (arr.Length == 2 && double.TryParse(arr[0], invNum, out var val)) { var slp = arr[1] switch { @@ -846,7 +849,7 @@ private void ProcessWh25(EcowittLocalApi.Wh25Sensor[] sensors, DateTime dateTime if (sensor.abs != null) { var arr = sensor.abs.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out var val)) + if (arr.Length == 2 && double.TryParse(arr[0], invNum, out var val)) { var abs = arr[1] switch { @@ -916,7 +919,7 @@ private void ProcessRain(EcowittLocalApi.CommonSensor[] sensors) try { var arr = sensor.val.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out var val)) + if (arr.Length == 2 && double.TryParse(arr[0], invNum, out var val)) { var evnt = arr[1] switch { @@ -942,7 +945,7 @@ private void ProcessRain(EcowittLocalApi.CommonSensor[] sensors) try { var arr = sensor.val.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out var val)) + if (arr.Length == 2 && double.TryParse(arr[0], invNum, out var val)) { var rate = arr[1] switch { @@ -968,7 +971,7 @@ private void ProcessRain(EcowittLocalApi.CommonSensor[] sensors) try { var arr = sensor.val.Split(' '); - if (arr.Length == 2 && double.TryParse(arr[0], out var val)) + if (arr.Length == 2 && double.TryParse(arr[0], invNum, out var val)) { var yr = arr[1] switch { diff --git a/CumulusMX/EcowittLocalApi.cs b/CumulusMX/EcowittLocalApi.cs index 505bbc1d..0dcf856d 100644 --- a/CumulusMX/EcowittLocalApi.cs +++ b/CumulusMX/EcowittLocalApi.cs @@ -11,7 +11,7 @@ namespace CumulusMX internal sealed class EcowittLocalApi : IDisposable { private readonly Cumulus cumulus; - private static NumberFormatInfo invNum = CultureInfo.InvariantCulture.NumberFormat; + private static readonly NumberFormatInfo invNum = CultureInfo.InvariantCulture.NumberFormat; public EcowittLocalApi(Cumulus cumul) { cumulus = cumul; From 0d23979d8c27e76486f07b6d5098ddcd476e1458 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Fri, 13 Sep 2024 14:46:10 +0100 Subject: [PATCH 37/63] More piezo srain logic added --- CumulusMX/EcowittHttpApiStation.cs | 25 ++++++++++++++++++++----- CumulusMX/HttpStationEcowitt.cs | 15 +++++++++------ 2 files changed, 29 insertions(+), 11 deletions(-) diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index df818c19..a825fe43 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -167,11 +167,14 @@ public override void Start() // process rain values if (cumulus.Gw1000PrimaryRainSensor == 0 && rawData.rain != null) { - ProcessRain(rawData.rain); + ProcessRain(rawData.rain, false); } - else if (cumulus.Gw1000PrimaryRainSensor == 1 && rawData.piezoRain != null) + + if ((cumulus.Gw1000PrimaryRainSensor == 1 || (cumulus.Gw1000PrimaryRainSensor == 0 && cumulus.EcowittIsRainingUsePiezo)) && rawData.piezoRain != null) { - ProcessRain(rawData.piezoRain); + // if we are using piezo as the primary rain sensor + // or using the tipper at the primary, but want to use the piezo srain value for IsRaining + ProcessRain(rawData.piezoRain, cumulus.Gw1000PrimaryRainSensor == 0 && cumulus.EcowittIsRainingUsePiezo); } if (rawData.lightning != null) @@ -875,7 +878,7 @@ private void ProcessWh25(EcowittLocalApi.Wh25Sensor[] sensors, DateTime dateTime } } - private void ProcessRain(EcowittLocalApi.CommonSensor[] sensors) + private void ProcessRain(EcowittLocalApi.CommonSensor[] sensors, bool isRainingOnly) { //"rain"/"piezoRain": [ // { @@ -916,6 +919,8 @@ private void ProcessRain(EcowittLocalApi.CommonSensor[] sensors) { case "0x0D": //Rain Event (val unit) + if (isRainingOnly) + break; try { var arr = sensor.val.Split(' '); @@ -942,6 +947,8 @@ private void ProcessRain(EcowittLocalApi.CommonSensor[] sensors) case "0x0E": //Rain Rate (val unit/h) + if (isRainingOnly) + break; try { var arr = sensor.val.Split(' '); @@ -958,6 +965,12 @@ private void ProcessRain(EcowittLocalApi.CommonSensor[] sensors) { rainRateLast = rate; } + + if (cumulus.StationOptions.UseRainForIsRaining == 1 && !cumulus.EcowittIsRainingUsePiezo) + { + IsRaining = rate > 0; + cumulus.IsRainingAlarm.Triggered = IsRaining; + } } } catch (Exception ex) @@ -968,6 +981,8 @@ private void ProcessRain(EcowittLocalApi.CommonSensor[] sensors) case "0x13": //Rain Year (val unit) + if (isRainingOnly) + break; try { var arr = sensor.val.Split(' '); @@ -996,7 +1011,7 @@ private void ProcessRain(EcowittLocalApi.CommonSensor[] sensors) if (cumulus.EcowittIsRainingUsePiezo) { IsRaining = sensor.val == "1"; - } + cumulus.IsRainingAlarm.Triggered = IsRaining; } break; case "0x10": // Rain day diff --git a/CumulusMX/HttpStationEcowitt.cs b/CumulusMX/HttpStationEcowitt.cs index 598f423b..6efeb113 100644 --- a/CumulusMX/HttpStationEcowitt.cs +++ b/CumulusMX/HttpStationEcowitt.cs @@ -627,11 +627,6 @@ public string ApplyData(string dataString, bool main, DateTime? ts = null) { rain = data["yearlyrainin"] ?? data["totalrainin"]; rRate = data["rainratein"]; - if (cumulus.StationOptions.UseRainForIsRaining == 2) - { - IsRaining = Convert.ToDouble(data["rrain_piezo"], invNum) > 0; - cumulus.IsRainingAlarm.Triggered = IsRaining; - } } else { @@ -639,6 +634,13 @@ public string ApplyData(string dataString, bool main, DateTime? ts = null) rRate = data["rrain_piezo"]; } + if (!cumulus.EcowittIsRainingUsePiezo) + { + IsRaining = (cumulus.StationOptions.UseRainForIsRaining == 0 ? Convert.ToDouble(rRate, invNum): Convert.ToDouble(data["rrain_piezo"], invNum)) > 0; + cumulus.IsRainingAlarm.Triggered = IsRaining; + } + + if (rRate == null) { // No rain rate, so we will calculate it @@ -662,9 +664,10 @@ public string ApplyData(string dataString, bool main, DateTime? ts = null) DoRain(rainVal, rateVal, recDate); } - if (data["srain_piezo"] != null && cumulus.EcowittIsRainingUsePiezo) + if (cumulus.EcowittIsRainingUsePiezo && data["srain_piezo"] != null) { IsRaining = data["srain_piezo"] == "1"; + cumulus.IsRainingAlarm.Triggered = IsRaining; } } catch (Exception ex) From 94a1609a4aa46f8458bed2a4526417a29ee71cef Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sat, 14 Sep 2024 18:05:03 +0100 Subject: [PATCH 38/63] - Add Soil moisture sensors 9-16 to Ecowitt HTTP API decode - Add Soil moisture sensors 9-16 to Ecowitt TCP API decode of sensor info --- CumulusMX/Cumulus.cs | 2 +- CumulusMX/CumulusMX.csproj | 2 +- CumulusMX/EcowittHttpApiStation.cs | 12 ++++++++++-- CumulusMX/GW1000Api.cs | 12 ++++++++++-- CumulusMX/GW1000Station.cs | 10 ++++++---- Updates.txt | 12 ++++++++++-- 6 files changed, 38 insertions(+), 12 deletions(-) diff --git a/CumulusMX/Cumulus.cs b/CumulusMX/Cumulus.cs index 6188c099..f46ff992 100644 --- a/CumulusMX/Cumulus.cs +++ b/CumulusMX/Cumulus.cs @@ -2313,7 +2313,7 @@ internal void RealtimeTimerTick(object sender, ElapsedEventArgs elapsedEventArgs if (!string.IsNullOrEmpty(RealtimeProgram)) { - if (!File.Exists(ProgramOptions.StartupTask)) + if (!File.Exists(RealtimeProgram)) { LogWarningMessage($"Warning: Realtime program '{RealtimeProgram}' does not exist"); } diff --git a/CumulusMX/CumulusMX.csproj b/CumulusMX/CumulusMX.csproj index 77e50499..68f015ef 100644 --- a/CumulusMX/CumulusMX.csproj +++ b/CumulusMX/CumulusMX.csproj @@ -55,7 +55,7 @@ - 4.2.0.4035-beta + 4.2.0.4036-beta Copyright © 2015-$([System.DateTime]::Now.ToString('yyyy')) Cumulus MX $(AssemblyName) diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index a825fe43..574892bb 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -563,7 +563,8 @@ private async Task GetSensorIds() name = "wh31ch" + (sensor.type - 5); goto case 1001; case int n when (n > 13 && n < 22): // wh51 - soil moisture (8 chan) - break; + name = "wh51ch" + (sensor.type - 13); + goto case 1001; case int n when (n > 21 && n < 26): // wh41 - pm2.5 (4 chan) name = "wh41ch" + (sensor.type - 21); goto case 1003; @@ -579,7 +580,7 @@ private async Task GetSensorIds() case 39: // wh45 - co2 name = "wh45"; goto case 1003; - case int n when (n > 39 && n < 48): // wh35 - leasf wet (8 chan) + case int n when (n > 39 && n < 48): // wh35 - leaf wet (8 chan) name = "wh35ch" + (sensor.type - 39); goto case 1003; case 48: // wh90 @@ -600,6 +601,13 @@ private async Task GetSensorIds() updateRate = 8000; } goto case 1003; + case int n when (n > 49 && n < 58): // wh51 - soil moisture (chan 9-16) + name = "wh51ch" + (sensor.type - 49 + 8); + goto case 1001; + case int n when (n > 57 && n < 66): // undocumented block of 8 sensors - possibly future temperature or t/h chans 9-16? + name = "unkownch" + (sensor.type - 57 + 8); + break; + case 1001: // battery type 1 (0=OK, 1=LOW) diff --git a/CumulusMX/GW1000Api.cs b/CumulusMX/GW1000Api.cs index 478cbdc3..541a32e6 100644 --- a/CumulusMX/GW1000Api.cs +++ b/CumulusMX/GW1000Api.cs @@ -374,7 +374,7 @@ private enum CommandRespSize internal enum SensorIds { - Wh65, // 0 00 + Wh69, // 0 00 Wh68, // 1 01 Wh80, // 2 02 Wh40, // 3 03 @@ -423,7 +423,15 @@ internal enum SensorIds Wh35Ch7, // 46 2E Wh35Ch8, // 47 2F Wh90, // 48 30 - Ws85 // 49 31 + Ws85, // 49 31 + Wh51Ch9, // 50 32 + Wh51Ch10, // 51 33 + Wh51Ch11, // 52 34 + Wh51Ch12, // 53 35 + Wh51Ch13, // 54 36 + Wh51Ch14, // 55 37 + Wh51Ch15, // 56 38 + Wh51Ch16 // 57 39 }; /* diff --git a/CumulusMX/GW1000Station.cs b/CumulusMX/GW1000Station.cs index 995a63b5..dcf1cdf2 100644 --- a/CumulusMX/GW1000Station.cs +++ b/CumulusMX/GW1000Station.cs @@ -741,10 +741,11 @@ private bool PrintSensorInfoNew(byte[] data, int idx) var battPos = idx + 5; var sigPos = idx + 6; // Wh65 could be a Wh65 or a Wh24, we found out using the System Info command - if (type == "WH65") - { - type = "WH24/WH65"; - } + // So, Ecowitt reused this id for the third time! It is now the WS69 + //if (type == "WH65") + //{ + // type = "WH24/WH65"; + //} switch (id) { @@ -787,6 +788,7 @@ private bool PrintSensorInfoNew(byte[] data, int idx) case "WH24": case "WH26": case "WH24/WH65": + case "WS69": batt = TestBattery1(data[battPos], 1); // 0 or 1 break; diff --git a/Updates.txt b/Updates.txt index dced2eec..45401d9e 100644 --- a/Updates.txt +++ b/Updates.txt @@ -1,5 +1,12 @@ -4.2.0 - b4035 +4.2.0 - b4036 ————————————— +BETA 4036 Changes: +—————————————————— +- Accessibility updates for the Interval and Daily data viewer pages +- Add Soil moisture sensors 9-16 to Ecowitt HTTP API decode +- Add Soil moisture sensors 9-16 to Ecowitt TCP API decode of sensor info + + BETA 4035 Changes: —————————————————— - Fixes to the Interval/Daily data viewer web pages @@ -47,7 +54,8 @@ New - New web tag <#MonthRainfall> which returns the rainfall total for the current month by default - Takes optional parameters y=YYYY and m=MM (both must be specified) to return the total rainfall for specified month in the specified year - Eg. <#MonthRainfall y=2018 m=10> -- Support for Ecowitt WS90 piezo IsRaining status to trigger MX IsRaining. Currently only supported with a WS90 connected to a GW2000 +- Support for Ecowitt WS90 piezo IsRaining status to trigger MX IsRaining. + - Currently only supported with a WS90/WS85 connected to a GW2000 (Sept. 2024). This value is being added to more stations as they get firmware updates. Changed - Davis WLL/Davis Cloud stations now use the WL.com subscription level to determine if they use WL.com as a logger From 22a885bfba16657b2c71ceca99ff4c8da6ec386a Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sat, 14 Sep 2024 22:12:09 +0100 Subject: [PATCH 39/63] Soil moisture units now follow the source --- CumulusMX/Cumulus.cs | 4 +- CumulusMX/EcowittCloudStation.cs | 80 +++++++++++++++++++++--------- CumulusMX/EcowittHttpApiStation.cs | 5 +- CumulusMX/GW1000Station.cs | 2 +- CumulusMX/HttpStationAmbient.cs | 14 ++++-- CumulusMX/HttpStationEcowitt.cs | 13 +++-- CumulusMX/HttpStationWund.cs | 2 +- CumulusMX/WeatherStation.cs | 36 +++++++------- Updates.txt | 4 ++ 9 files changed, 102 insertions(+), 58 deletions(-) diff --git a/CumulusMX/Cumulus.cs b/CumulusMX/Cumulus.cs index f46ff992..76469132 100644 --- a/CumulusMX/Cumulus.cs +++ b/CumulusMX/Cumulus.cs @@ -13554,14 +13554,14 @@ public class StationUnits public string PressTrendText { get; set; } public string WindRunText { get; set; } public string AirQualityUnitText { get; set; } - public string SoilMoistureUnitText { get; set; } + public string[] SoilMoistureUnitText { get; set; } = new string[16]; public string CO2UnitText { get; set; } public string LeafWetnessUnitText { get; set; } public StationUnits() { AirQualityUnitText = "µg/m³"; - SoilMoistureUnitText = "cb"; + Array.Fill(SoilMoistureUnitText, "cb"); CO2UnitText = "ppm"; LeafWetnessUnitText = string.Empty; // Davis is unitless, Ecowitt uses % } diff --git a/CumulusMX/EcowittCloudStation.cs b/CumulusMX/EcowittCloudStation.cs index 26eb46cb..ce7a1915 100644 --- a/CumulusMX/EcowittCloudStation.cs +++ b/CumulusMX/EcowittCloudStation.cs @@ -12,7 +12,7 @@ internal class EcowittCloudStation : WeatherStation private readonly EcowittApi ecowittApi; private int maxArchiveRuns = 1; private Task liveTask; - private readonly bool main; + private readonly bool mainStation; private string deviceModel; private Version deviceFirmware; private int lastHour = -1; @@ -21,9 +21,9 @@ public EcowittCloudStation(Cumulus cumulus, WeatherStation station = null) : bas { this.station = station; - main = station == null; + mainStation = station == null; - if (main) + if (mainStation) { cumulus.LogMessage("Creating Ecowitt Cloud Station"); } @@ -40,7 +40,7 @@ public EcowittCloudStation(Cumulus cumulus, WeatherStation station = null) : bas // Do not set these if we are only using extra sensors - if (main) + if (mainStation) { // cloud provides 10 min average wind speeds cumulus.StationOptions.CalcuateAverageWindSpeed = true; @@ -90,15 +90,15 @@ public EcowittCloudStation(Cumulus cumulus, WeatherStation station = null) : bas DataTimeoutMins = 2; } - if (main || cumulus.EcowittExtraUseAQI) + if (mainStation || cumulus.EcowittExtraUseAQI) { cumulus.Units.AirQualityUnitText = "µg/m³"; } - if (main || cumulus.EcowittExtraUseSoilMoist) + if (mainStation) { - cumulus.Units.SoilMoistureUnitText = "%"; + Array.Fill(cumulus.Units.SoilMoistureUnitText, "%"); } - if (main || cumulus.EcowittExtraUseSoilMoist) + if (mainStation || cumulus.EcowittExtraUseLeafWet) { cumulus.Units.LeafWetnessUnitText = "%"; } @@ -106,7 +106,7 @@ public EcowittCloudStation(Cumulus cumulus, WeatherStation station = null) : bas ecowittApi = new EcowittApi(cumulus, this); // Only perform the Start-up if we are a proper station, not a Extra Sensor - if (main) + if (mainStation) { Task.Run(getAndProcessHistoryData); var retVal = ecowittApi.GetStationList(true, cumulus.EcowittMacAddress, cumulus.cancellationToken); @@ -178,7 +178,7 @@ public override void Start() if (hour == 13) { - var retVal = ecowittApi.GetStationList(main || cumulus.EcowittExtraUseCamera, cumulus.EcowittMacAddress, cumulus.cancellationToken); + var retVal = ecowittApi.GetStationList(mainStation || cumulus.EcowittExtraUseCamera, cumulus.EcowittMacAddress, cumulus.cancellationToken); if (retVal.Length == 2 && !retVal[1].StartsWith("EasyWeather")) { // EasyWeather seems to contain the WiFi version @@ -239,7 +239,7 @@ public override void getAndProcessHistoryData() public override string GetEcowittCameraUrl() { - if ((cumulus.EcowittExtraUseCamera || main) && !string.IsNullOrEmpty(cumulus.EcowittCameraMacAddress)) + if ((cumulus.EcowittExtraUseCamera || mainStation) && !string.IsNullOrEmpty(cumulus.EcowittCameraMacAddress)) { try { @@ -257,7 +257,7 @@ public override string GetEcowittCameraUrl() public override string GetEcowittVideoUrl() { - if ((cumulus.EcowittExtraUseCamera || main) && !string.IsNullOrEmpty(cumulus.EcowittCameraMacAddress)) + if ((cumulus.EcowittExtraUseCamera || mainStation) && !string.IsNullOrEmpty(cumulus.EcowittCameraMacAddress)) { try { @@ -295,13 +295,13 @@ private void GetHistoricData() private void ProcessCurrentData(EcowittApi.CurrentDataData data, CancellationToken token) { bool batteryLow = false; - var thisStation = main ? this : station; + var thisStation = mainStation ? this : station; token.ThrowIfCancellationRequested(); try { // Only do the primary sensors if running as the main station - if (main) + if (mainStation) { // Outdoor temp/hum if (cumulus.Gw1000PrimaryTHSensor == 0) @@ -445,7 +445,7 @@ private void ProcessCurrentData(EcowittApi.CurrentDataData data, CancellationTok } // Solar - if ((main || cumulus.EcowittExtraUseSolar) && data.solar_and_uvi != null) + if ((mainStation || cumulus.EcowittExtraUseSolar) && data.solar_and_uvi != null) { try { @@ -462,7 +462,7 @@ private void ProcessCurrentData(EcowittApi.CurrentDataData data, CancellationTok } // Extra Temperature - if (main || cumulus.EcowittExtraUseTempHum) + if (mainStation || cumulus.EcowittExtraUseTempHum) { try { @@ -475,7 +475,7 @@ private void ProcessCurrentData(EcowittApi.CurrentDataData data, CancellationTok } // === Soil/Water Temp === - if (main || cumulus.EcowittExtraUseUserTemp) + if (mainStation || cumulus.EcowittExtraUseUserTemp) { try { @@ -488,7 +488,7 @@ private void ProcessCurrentData(EcowittApi.CurrentDataData data, CancellationTok } // === Soil Moisture === - if (main || cumulus.EcowittExtraUseSoilMoist) + if (mainStation || cumulus.EcowittExtraUseSoilMoist) { try { @@ -501,7 +501,7 @@ private void ProcessCurrentData(EcowittApi.CurrentDataData data, CancellationTok } // === Leaf Wetness === - if (main || cumulus.EcowittExtraUseLeafWet) + if (mainStation || cumulus.EcowittExtraUseLeafWet) { try { @@ -514,7 +514,7 @@ private void ProcessCurrentData(EcowittApi.CurrentDataData data, CancellationTok } // === Air Quality === - if (main || cumulus.EcowittExtraUseAQI) + if (mainStation || cumulus.EcowittExtraUseAQI) { try { @@ -527,7 +527,7 @@ private void ProcessCurrentData(EcowittApi.CurrentDataData data, CancellationTok } // === CO₂ === - if (main || cumulus.EcowittExtraUseCo2) + if (mainStation || cumulus.EcowittExtraUseCo2) { try { @@ -540,7 +540,7 @@ private void ProcessCurrentData(EcowittApi.CurrentDataData data, CancellationTok } // === Lightning === - if (main || cumulus.EcowittExtraUseLightning) + if (mainStation || cumulus.EcowittExtraUseLightning) { try { @@ -553,7 +553,7 @@ private void ProcessCurrentData(EcowittApi.CurrentDataData data, CancellationTok } // === Leak === - if (main || cumulus.EcowittExtraUseLeak) + if (mainStation || cumulus.EcowittExtraUseLeak) { try { @@ -871,46 +871,78 @@ private void ProcessUserTemps(EcowittApi.CurrentDataData data, WeatherStation st } } - private static void ProcessSoilMoist(EcowittApi.CurrentDataData data, WeatherStation station) + private void ProcessSoilMoist(EcowittApi.CurrentDataData data, WeatherStation station) { if (data.soil_ch1 != null) { station.DoSoilMoisture(data.soil_ch1.soilmoisture.value, 1); + if (!mainStation) + { + cumulus.Units.SoilMoistureUnitText[0] = "%"; + } } if (data.soil_ch2 != null) { station.DoSoilMoisture(data.soil_ch2.soilmoisture.value, 2); + if (!mainStation) + { + cumulus.Units.SoilMoistureUnitText[1] = "%"; + } } if (data.soil_ch3 != null) { station.DoSoilMoisture(data.soil_ch3.soilmoisture.value, 3); + if (!mainStation) + { + cumulus.Units.SoilMoistureUnitText[2] = "%"; + } } if (data.soil_ch4 != null) { station.DoSoilMoisture(data.soil_ch4.soilmoisture.value, 4); + if (!mainStation) + { + cumulus.Units.SoilMoistureUnitText[3] = "%"; + } } if (data.soil_ch5 != null) { station.DoSoilMoisture(data.soil_ch5.soilmoisture.value, 5); + if (!mainStation) + { + cumulus.Units.SoilMoistureUnitText[4] = "%"; + } } if (data.soil_ch6 != null) { station.DoSoilMoisture(data.soil_ch6.soilmoisture.value, 6); + if (!mainStation) + { + cumulus.Units.SoilMoistureUnitText[5] = "%"; + } } if (data.soil_ch7 != null) { station.DoSoilMoisture(data.soil_ch7.soilmoisture.value, 7); + if (!mainStation) + { + cumulus.Units.SoilMoistureUnitText[6] = "%"; + } } if (data.soil_ch8 != null) { station.DoSoilMoisture(data.soil_ch8.soilmoisture.value, 8); + if (!mainStation) + { + cumulus.Units.SoilMoistureUnitText[7] = "%"; + } } } diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index 574892bb..52bec6ad 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -52,7 +52,7 @@ internal class EcowittHttpApiStation : WeatherStation public EcowittHttpApiStation(Cumulus cumulus) : base(cumulus) { cumulus.Units.AirQualityUnitText = "µg/m³"; - cumulus.Units.SoilMoistureUnitText = "%"; + Array.Fill(cumulus.Units.SoilMoistureUnitText, "%"); cumulus.Units.LeafWetnessUnitText = "%"; // GW1000 does not provide 10 min average wind speeds @@ -604,9 +604,6 @@ private async Task GetSensorIds() case int n when (n > 49 && n < 58): // wh51 - soil moisture (chan 9-16) name = "wh51ch" + (sensor.type - 49 + 8); goto case 1001; - case int n when (n > 57 && n < 66): // undocumented block of 8 sensors - possibly future temperature or t/h chans 9-16? - name = "unkownch" + (sensor.type - 57 + 8); - break; diff --git a/CumulusMX/GW1000Station.cs b/CumulusMX/GW1000Station.cs index dcf1cdf2..29733a9e 100644 --- a/CumulusMX/GW1000Station.cs +++ b/CumulusMX/GW1000Station.cs @@ -44,7 +44,7 @@ internal class GW1000Station : WeatherStation public GW1000Station(Cumulus cumulus) : base(cumulus) { cumulus.Units.AirQualityUnitText = "µg/m³"; - cumulus.Units.SoilMoistureUnitText = "%"; + Array.Fill(cumulus.Units.SoilMoistureUnitText, "%"); cumulus.Units.LeafWetnessUnitText = "%"; // GW1000 does not provide 10 min average wind speeds diff --git a/CumulusMX/HttpStationAmbient.cs b/CumulusMX/HttpStationAmbient.cs index fb4a50d1..d23553e3 100644 --- a/CumulusMX/HttpStationAmbient.cs +++ b/CumulusMX/HttpStationAmbient.cs @@ -10,12 +10,14 @@ namespace CumulusMX class HttpStationAmbient : WeatherStation { private readonly WeatherStation station; + private readonly Cumulus cumulus; private bool starting = true; private bool stopping = false; public HttpStationAmbient(Cumulus cumulus, WeatherStation station = null) : base(cumulus, station != null) { this.station = station; + this.cumulus = cumulus; if (station == null) { @@ -34,13 +36,13 @@ public HttpStationAmbient(Cumulus cumulus, WeatherStation station = null) : base // Ambient does not send DP, so force MX to calculate it //cumulus.StationOptions.CalculatedDP = true - if (station == null || (station != null && cumulus.AmbientExtraUseAQI)) + if (station == null || cumulus.AmbientExtraUseAQI) { cumulus.Units.AirQualityUnitText = "µg/m³"; } - if (station == null || (station != null && cumulus.AmbientExtraUseSoilMoist)) + if (station == null) { - cumulus.Units.SoilMoistureUnitText = "%"; + Array.Fill(cumulus.Units.SoilMoistureUnitText, "%"); } // Only perform the Start-up if we are a proper station, not a Extra Sensor @@ -677,13 +679,17 @@ private static void ProcessSoilTemps(NameValueCollection data, WeatherStation st } } - private static void ProcessSoilMoist(NameValueCollection data, WeatherStation station) + private void ProcessSoilMoist(NameValueCollection data, WeatherStation station) { for (var i = 1; i <= 10; i++) { if (data["soilhum" + i] != null) { station.DoSoilMoisture(Convert.ToDouble(data["soilhum" + i], CultureInfo.InvariantCulture), i); + if (station != null) + { + cumulus.Units.SoilMoistureUnitText[i - 1] = "%"; + } } } } diff --git a/CumulusMX/HttpStationEcowitt.cs b/CumulusMX/HttpStationEcowitt.cs index 6efeb113..5c829ed4 100644 --- a/CumulusMX/HttpStationEcowitt.cs +++ b/CumulusMX/HttpStationEcowitt.cs @@ -16,6 +16,7 @@ namespace CumulusMX partial class HttpStationEcowitt : WeatherStation { private readonly WeatherStation station; + private readonly bool mainStation; private bool starting = true; private bool stopping = false; private readonly NumberFormatInfo invNum = CultureInfo.InvariantCulture.NumberFormat; @@ -32,7 +33,7 @@ public HttpStationEcowitt(Cumulus cumulus, WeatherStation station = null) : base { this.station = station; - var mainStation = station == null; + mainStation = station == null; if (mainStation) { @@ -115,11 +116,11 @@ public HttpStationEcowitt(Cumulus cumulus, WeatherStation station = null) : base { cumulus.Units.AirQualityUnitText = "µg/m³"; } - if (mainStation || cumulus.EcowittExtraUseSoilMoist) + if (mainStation) { - cumulus.Units.SoilMoistureUnitText = "%"; + Array.Fill(cumulus.Units.SoilMoistureUnitText, "%"); } - if (mainStation || cumulus.EcowittExtraUseSoilMoist) + if (mainStation || cumulus.EcowittExtraUseLeafWet) { cumulus.Units.LeafWetnessUnitText = "%"; } @@ -1191,6 +1192,10 @@ private void ProcessSoilMoist(NameValueCollection data, WeatherStation station) if (data["soilmoisture" + i] != null) { station.DoSoilMoisture(Convert.ToDouble(data["soilmoisture" + i], invNum), i); + if (!mainStation) + { + cumulus.Units.SoilMoistureUnitText[i - 1] = "%"; + } } } } diff --git a/CumulusMX/HttpStationWund.cs b/CumulusMX/HttpStationWund.cs index 924fbc28..2fb43d08 100644 --- a/CumulusMX/HttpStationWund.cs +++ b/CumulusMX/HttpStationWund.cs @@ -18,7 +18,7 @@ public HttpStationWund(Cumulus cumulus) : base(cumulus) cumulus.StationOptions.CalculatedWC = true; cumulus.Units.AirQualityUnitText = "µg/m³"; - cumulus.Units.SoilMoistureUnitText = "%"; + Array.Fill(cumulus.Units.SoilMoistureUnitText, "%"); cumulus.Units.LeafWetnessUnitText = "%"; // Wunderground does not send the rain rate, so we will calculate it diff --git a/CumulusMX/WeatherStation.cs b/CumulusMX/WeatherStation.cs index c4d12786..86f0b72b 100644 --- a/CumulusMX/WeatherStation.cs +++ b/CumulusMX/WeatherStation.cs @@ -11335,37 +11335,37 @@ public string GetSoilMoisture() var json = new StringBuilder("{\"data\":[", 1024); if (cumulus.GraphOptions.Visible.SoilMoist.ValVisible(0, true)) - json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[0]}\",\"{SoilMoisture1:F0}\",\"{cumulus.Units.SoilMoistureUnitText}\"],"); + json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[0]}\",\"{SoilMoisture1:F0}\",\"{cumulus.Units.SoilMoistureUnitText[0]}\"],"); if (cumulus.GraphOptions.Visible.SoilMoist.ValVisible(1, true)) - json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[1]}\",\"{SoilMoisture2:F0}\",\"{cumulus.Units.SoilMoistureUnitText}\"],"); + json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[1]}\",\"{SoilMoisture2:F0}\",\"{cumulus.Units.SoilMoistureUnitText[1]}\"],"); if (cumulus.GraphOptions.Visible.SoilMoist.ValVisible(2, true)) - json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[2]}\",\"{SoilMoisture3:F0}\",\"{cumulus.Units.SoilMoistureUnitText}\"],"); + json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[2]}\",\"{SoilMoisture3:F0}\",\"{cumulus.Units.SoilMoistureUnitText[2]}\"],"); if (cumulus.GraphOptions.Visible.SoilMoist.ValVisible(3, true)) - json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[3]}\",\"{SoilMoisture4:F0}\",\"{cumulus.Units.SoilMoistureUnitText}\"],"); + json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[3]}\",\"{SoilMoisture4:F0}\",\"{cumulus.Units.SoilMoistureUnitText[3]}\"],"); if (cumulus.GraphOptions.Visible.SoilMoist.ValVisible(4, true)) - json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[4]}\",\"{SoilMoisture5:F0}\",\"{cumulus.Units.SoilMoistureUnitText}\"],"); + json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[4]}\",\"{SoilMoisture5:F0}\",\"{cumulus.Units.SoilMoistureUnitText[4]}\"],"); if (cumulus.GraphOptions.Visible.SoilMoist.ValVisible(5, true)) - json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[5]}\",\"{SoilMoisture6:F0}\",\"{cumulus.Units.SoilMoistureUnitText}\"],"); + json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[5]}\",\"{SoilMoisture6:F0}\",\"{cumulus.Units.SoilMoistureUnitText[5]}\"],"); if (cumulus.GraphOptions.Visible.SoilMoist.ValVisible(6, true)) - json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[6]}\",\"{SoilMoisture7:F0}\",\"{cumulus.Units.SoilMoistureUnitText}\"],"); + json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[6]}\",\"{SoilMoisture7:F0}\",\"{cumulus.Units.SoilMoistureUnitText[6]}\"],"); if (cumulus.GraphOptions.Visible.SoilMoist.ValVisible(7, true)) - json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[7]}\",\"{SoilMoisture8:F0}\",\"{cumulus.Units.SoilMoistureUnitText}\"],"); + json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[7]}\",\"{SoilMoisture8:F0}\",\"{cumulus.Units.SoilMoistureUnitText[7]}\"],"); if (cumulus.GraphOptions.Visible.SoilMoist.ValVisible(8, true)) - json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[8]}\",\"{SoilMoisture9:F0}\",\"{cumulus.Units.SoilMoistureUnitText}\"],"); + json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[8]}\",\"{SoilMoisture9:F0}\",\"{cumulus.Units.SoilMoistureUnitText[8]}\"],"); if (cumulus.GraphOptions.Visible.SoilMoist.ValVisible(9, true)) - json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[9]}\",\"{SoilMoisture10:F0}\",\"{cumulus.Units.SoilMoistureUnitText}\"],"); + json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[9]}\",\"{SoilMoisture10:F0}\",\"{cumulus.Units.SoilMoistureUnitText[9]}\"],"); if (cumulus.GraphOptions.Visible.SoilMoist.ValVisible(10, true)) - json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[10]}\",\"{SoilMoisture11:F0}\",\"{cumulus.Units.SoilMoistureUnitText}\"],"); + json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[10]}\",\"{SoilMoisture11:F0}\",\"{cumulus.Units.SoilMoistureUnitText[10]}\"],"); if (cumulus.GraphOptions.Visible.SoilMoist.ValVisible(11, true)) - json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[11]}\",\"{SoilMoisture12:F0}\",\"{cumulus.Units.SoilMoistureUnitText}\"],"); + json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[11]}\",\"{SoilMoisture12:F0}\",\"{cumulus.Units.SoilMoistureUnitText[11]}\"],"); if (cumulus.GraphOptions.Visible.SoilMoist.ValVisible(12, true)) - json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[12]}\",\"{SoilMoisture13:F0}\",\"{cumulus.Units.SoilMoistureUnitText}\"],"); + json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[12]}\",\"{SoilMoisture13:F0}\",\"{cumulus.Units.SoilMoistureUnitText[12]}\"],"); if (cumulus.GraphOptions.Visible.SoilMoist.ValVisible(13, true)) - json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[13]}\",\"{SoilMoisture14:F0}\",\"{cumulus.Units.SoilMoistureUnitText}\"],"); + json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[13]}\",\"{SoilMoisture14:F0}\",\"{cumulus.Units.SoilMoistureUnitText[13]}\"],"); if (cumulus.GraphOptions.Visible.SoilMoist.ValVisible(14, true)) - json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[14]}\",\"{SoilMoisture15:F0}\",\"{cumulus.Units.SoilMoistureUnitText}\"],"); + json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[14]}\",\"{SoilMoisture15:F0}\",\"{cumulus.Units.SoilMoistureUnitText[14]}\"],"); if (cumulus.GraphOptions.Visible.SoilMoist.ValVisible(15, true)) - json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[15]}\",\"{SoilMoisture16:F0}\",\"{cumulus.Units.SoilMoistureUnitText}\"]"); + json.Append($"[\"{cumulus.Trans.SoilMoistureCaptions[15]}\",\"{SoilMoisture16:F0}\",\"{cumulus.Units.SoilMoistureUnitText[15]}\"]"); if (json[^1] == ',') json.Length--; @@ -12590,7 +12590,7 @@ public string GetUnits() json.Append($"\"windrun\":\"{cumulus.Units.WindRunText}\","); json.Append($"\"rain\":\"{cumulus.Units.RainText}\","); json.Append($"\"press\":\"{cumulus.Units.PressText}\","); - json.Append($"\"soilmoisture\":\"{cumulus.Units.SoilMoistureUnitText}\","); + json.Append($"\"soilmoisture\":[\"{string.Join("\",\"", cumulus.Units.SoilMoistureUnitText)}\"],"); json.Append($"\"co2\":\"{cumulus.Units.CO2UnitText}\","); json.Append($"\"leafwet\":\"{cumulus.Units.LeafWetnessUnitText}\","); json.Append($"\"aq\":\"{cumulus.Units.AirQualityUnitText}\""); @@ -12608,7 +12608,7 @@ public string GetGraphConfig(bool local) json.Append($"\"press\":{{\"units\":\"{cumulus.Units.PressText}\",\"decimals\":{cumulus.PressDPlaces}}},"); json.Append($"\"hum\":{{\"decimals\":{cumulus.HumDPlaces}}},"); json.Append($"\"uv\":{{\"decimals\":{cumulus.UVDPlaces}}},"); - json.Append($"\"soilmoisture\":{{\"units\":\"{cumulus.Units.SoilMoistureUnitText}\"}},"); + json.Append($"\"soilmoisture\":{{\"units\":[\"{string.Join("\",\"", cumulus.Units.SoilMoistureUnitText)}\"]}},"); json.Append($"\"co2\":{{\"units\":\"{cumulus.Units.CO2UnitText}\"}},"); json.Append($"\"leafwet\":{{\"units\":\"{cumulus.Units.LeafWetnessUnitText}\",\"decimals\":{cumulus.LeafWetDPlaces}}},"); json.Append($"\"aq\":{{\"units\":\"{cumulus.Units.AirQualityUnitText}\"}},"); diff --git a/Updates.txt b/Updates.txt index 45401d9e..dd8c3de4 100644 --- a/Updates.txt +++ b/Updates.txt @@ -5,6 +5,7 @@ BETA 4036 Changes: - Accessibility updates for the Interval and Daily data viewer pages - Add Soil moisture sensors 9-16 to Ecowitt HTTP API decode - Add Soil moisture sensors 9-16 to Ecowitt TCP API decode of sensor info +- Soil moisture units now follow the source BETA 4035 Changes: @@ -69,6 +70,9 @@ Fixed - Davis Cloud stations no longer continuously try to fetch history data if there is no Pro subscription - WMR928 Station now correctly converts indoor temperatures to the user defined units - Not logging PHP upload failures to the warning log +- Soil moisture units now follow the source + - Example if you have a main station Davis with sensors 1 & 2, their units will be cb, and you have Ecowitt extra sensors 3 & 4, their units will be % + - This does mean a change to the Units JSON and the graph scripts for the default web site. A re-upload of /js/cumuluscharts.js and /js/selectachart.js will be required Package Updates From a20c9835add3e490b260a3a5d3c5c54a5ae2ba0d Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sun, 15 Sep 2024 10:44:14 +0100 Subject: [PATCH 40/63] Two new web tags <#NewRecordAlarm> and <#NewRecordAlarmMessage> --- CumulusMX/webtags.cs | 17 +++++++++++++++++ Updates.txt | 5 +++++ 2 files changed, 22 insertions(+) diff --git a/CumulusMX/webtags.cs b/CumulusMX/webtags.cs index 8bfdb3de..9db65a31 100644 --- a/CumulusMX/webtags.cs +++ b/CumulusMX/webtags.cs @@ -4571,6 +4571,21 @@ private string TagFirmwareAlarm(Dictionary tagParams) return "0"; } + private string TagNewRecordAlarm(Dictionary tagParams) + { + if (cumulus.NewRecordAlarm.Enabled) + { + return cumulus.NewRecordAlarm.Triggered ? "1" : "0"; + } + + return "0"; + } + + private string TagNewRecordAlarmMessage(Dictionary tagParams) + { + return cumulus.NewRecordAlarm.LastMessage; + } + private string TagMySqlUploadAlarm(Dictionary tagParams) { @@ -6479,6 +6494,8 @@ public void InitialiseWebtags() { "HttpUploadAlarm", TagHttpUploadAlarm }, { "UpgradeAlarm", TagUpgradeAlarm }, { "FirmwareAlarm", TagFirmwareAlarm }, + { "NewRecordAlarm", TagNewRecordAlarm }, + { "NewRecordAlarmMessage", TagNewRecordAlarmMessage }, { "RG11RainToday", TagRg11RainToday }, { "RG11RainYest", TagRg11RainYest }, diff --git a/Updates.txt b/Updates.txt index dd8c3de4..21d1d25e 100644 --- a/Updates.txt +++ b/Updates.txt @@ -6,6 +6,8 @@ BETA 4036 Changes: - Add Soil moisture sensors 9-16 to Ecowitt HTTP API decode - Add Soil moisture sensors 9-16 to Ecowitt TCP API decode of sensor info - Soil moisture units now follow the source +- Two new web tags <#NewRecordAlarm> and <#NewRecordAlarmMessage> + BETA 4035 Changes: @@ -57,6 +59,9 @@ New - Eg. <#MonthRainfall y=2018 m=10> - Support for Ecowitt WS90 piezo IsRaining status to trigger MX IsRaining. - Currently only supported with a WS90/WS85 connected to a GW2000 (Sept. 2024). This value is being added to more stations as they get firmware updates. +- Two new web tags <#NewRecordAlarm> and <#NewRecordAlarmMessage> + - NewRecordAlarm somewhat replicates the existing #newrecord web tag, but is also controlled by the alarm being enable/disabled + - NewRecordAlarmMessage displays the last new record alarm text message Changed - Davis WLL/Davis Cloud stations now use the WL.com subscription level to determine if they use WL.com as a logger From 1580fdefbe0b5ef37cbb694e740a1b84397832fa Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sun, 15 Sep 2024 14:53:22 +0100 Subject: [PATCH 41/63] Cleanup old MD5 hash files --- CumulusMX/Cumulus.cs | 13 +++++++++++++ Updates.txt | 3 ++- 2 files changed, 15 insertions(+), 1 deletion(-) diff --git a/CumulusMX/Cumulus.cs b/CumulusMX/Cumulus.cs index 76469132..5b5e43a9 100644 --- a/CumulusMX/Cumulus.cs +++ b/CumulusMX/Cumulus.cs @@ -643,6 +643,9 @@ public void Initialise(int HTTPport, bool DebugEnabled, string startParms) // Check if all the folders required by CMX exist, if not create them CreateRequiredFolders(); + // Remove old MD5 hash files + CleanUpHashFiles(); + Datapath = "data" + DirectorySeparator; backupPath = "backup" + DirectorySeparator; ReportPath = "Reports" + DirectorySeparator; @@ -3093,6 +3096,16 @@ private static List ParseParams(string line) public string DecimalSeparator { get; set; } + private void CleanUpHashFiles() + { + foreach (var file in Directory.EnumerateFiles(AppDir, "hash_md5_*.txt")) + { + if (!file.EndsWith(Build + ".txt")) + { + File.Delete(file); + } + } + } internal void DoMoonPhase() { diff --git a/Updates.txt b/Updates.txt index 21d1d25e..60d0a49a 100644 --- a/Updates.txt +++ b/Updates.txt @@ -7,7 +7,7 @@ BETA 4036 Changes: - Add Soil moisture sensors 9-16 to Ecowitt TCP API decode of sensor info - Soil moisture units now follow the source - Two new web tags <#NewRecordAlarm> and <#NewRecordAlarmMessage> - +- Cleanup old MD5 hash files BETA 4035 Changes: @@ -62,6 +62,7 @@ New - Two new web tags <#NewRecordAlarm> and <#NewRecordAlarmMessage> - NewRecordAlarm somewhat replicates the existing #newrecord web tag, but is also controlled by the alarm being enable/disabled - NewRecordAlarmMessage displays the last new record alarm text message +- Old MD5 hash files are now deleted on startup Changed - Davis WLL/Davis Cloud stations now use the WL.com subscription level to determine if they use WL.com as a logger From 97af9bade9200bde0dd93150f5228b7eea7c820f Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sun, 15 Sep 2024 15:14:08 +0100 Subject: [PATCH 42/63] More stations no update LastDataReadTime --- CumulusMX/DavisCloudStation.cs | 4 ++-- CumulusMX/DavisStation.cs | 12 ++++++------ CumulusMX/DavisWllStation.cs | 4 ++-- CumulusMX/EcowittApi.cs | 2 +- CumulusMX/EcowittHttpApiStation.cs | 1 + CumulusMX/GW1000Station.cs | 1 + CumulusMX/HttpStationAmbient.cs | 1 + CumulusMX/HttpStationEcowitt.cs | 1 + CumulusMX/JsonStation.cs | 1 + CumulusMX/WeatherStation.cs | 2 +- 10 files changed, 17 insertions(+), 12 deletions(-) diff --git a/CumulusMX/DavisCloudStation.cs b/CumulusMX/DavisCloudStation.cs index 58d2936e..8b791353 100644 --- a/CumulusMX/DavisCloudStation.cs +++ b/CumulusMX/DavisCloudStation.cs @@ -457,8 +457,8 @@ private void GetHistoricData(BackgroundWorker worker) cumulus.LogDebugMessage($"WeatherLink URL = {historicUrl.ToString().Replace(cumulus.WllApiKey, "API_KEY")}"); - lastDataReadTime = cumulus.LastUpdateTime; - int luhour = lastDataReadTime.Hour; + LastDataReadTime = cumulus.LastUpdateTime; + int luhour = LastDataReadTime.Hour; int rollHour = Math.Abs(cumulus.GetHourInc(lastHistoricData)); diff --git a/CumulusMX/DavisStation.cs b/CumulusMX/DavisStation.cs index aa7f7dcf..941b775b 100644 --- a/CumulusMX/DavisStation.cs +++ b/CumulusMX/DavisStation.cs @@ -2240,10 +2240,10 @@ private void GetArchiveData() NetworkStream stream = null; - lastDataReadTime = cumulus.LastUpdateTime; - int luhour = lastDataReadTime.Hour; + LastDataReadTime = cumulus.LastUpdateTime; + int luhour = LastDataReadTime.Hour; - int rollHour = Math.Abs(cumulus.GetHourInc(lastDataReadTime)); + int rollHour = Math.Abs(cumulus.GetHourInc(LastDataReadTime)); cumulus.LogMessage("GetArchiveData: Roll-over hour = " + rollHour); @@ -2573,7 +2573,7 @@ private void GetArchiveData() cumulus.LogMessage("GetArchiveData: Loaded archive record for Page=" + p + " Record=" + r + " Timestamp=" + archiveData.Timestamp); - if (timestamp > lastDataReadTime) + if (timestamp > LastDataReadTime) { cumulus.LogMessage("GetArchiveData: Processing archive record for " + timestamp); @@ -2592,7 +2592,7 @@ private void GetArchiveData() } else { - interval = (int) (timestamp - lastDataReadTime).TotalMinutes; + interval = (int) (timestamp - LastDataReadTime).TotalMinutes; } // ..and then process it @@ -2905,7 +2905,7 @@ private void GetArchiveData() DoFeelsLike(timestamp); DoHumidex(timestamp); - lastDataReadTime = timestamp; + LastDataReadTime = timestamp; _ = cumulus.DoLogFile(timestamp, false); cumulus.LogMessage("GetArchiveData: Log file entry written"); diff --git a/CumulusMX/DavisWllStation.cs b/CumulusMX/DavisWllStation.cs index d137cf09..7069e1d2 100644 --- a/CumulusMX/DavisWllStation.cs +++ b/CumulusMX/DavisWllStation.cs @@ -1544,8 +1544,8 @@ private void GetWlHistoricData(BackgroundWorker worker) cumulus.LogDebugMessage($"WeatherLink URL = {historicUrl.ToString().Replace(cumulus.WllApiKey, "API_KEY")}"); - lastDataReadTime = lastHistoricData; - int luhour = lastDataReadTime.Hour; + LastDataReadTime = lastHistoricData; + int luhour = LastDataReadTime.Hour; int rollHour = Math.Abs(cumulus.GetHourInc(lastHistoricData)); diff --git a/CumulusMX/EcowittApi.cs b/CumulusMX/EcowittApi.cs index 07fc8e60..fc6647b4 100644 --- a/CumulusMX/EcowittApi.cs +++ b/CumulusMX/EcowittApi.cs @@ -1410,7 +1410,7 @@ private void ProcessHistoryData(EcowittHistoricData data, CancellationToken toke station.DoTrendValues(rec.Key); station.UpdateStatusPanel(rec.Key); cumulus.AddToWebServiceLists(rec.Key); - + station.LastDataReadTime = rec.Key; } } diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index 52bec6ad..bac91f1d 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -277,6 +277,7 @@ public override void Start() dataReceived = true; DataStopped = false; cumulus.DataStoppedAlarm.Triggered = false; + LastDataReadTime = dataLastRead; var minute = DateTime.Now.Minute; if (minute != lastMinute) diff --git a/CumulusMX/GW1000Station.cs b/CumulusMX/GW1000Station.cs index 29733a9e..67b76e2e 100644 --- a/CumulusMX/GW1000Station.cs +++ b/CumulusMX/GW1000Station.cs @@ -1402,6 +1402,7 @@ private void GetLiveData() UpdateStatusPanel(dateTime); UpdateMQTT(); + LastDataReadTime = dateTime; dataReceived = true; DataStopped = false; diff --git a/CumulusMX/HttpStationAmbient.cs b/CumulusMX/HttpStationAmbient.cs index d23553e3..109fc74a 100644 --- a/CumulusMX/HttpStationAmbient.cs +++ b/CumulusMX/HttpStationAmbient.cs @@ -615,6 +615,7 @@ battrain Rain Gauge [0 or 1] thisStation.UpdateStatusPanel(recDate); thisStation.UpdateMQTT(); + thisStation.LastDataReadTime = recDate; } catch (Exception ex) { diff --git a/CumulusMX/HttpStationEcowitt.cs b/CumulusMX/HttpStationEcowitt.cs index 5c829ed4..567f38c1 100644 --- a/CumulusMX/HttpStationEcowitt.cs +++ b/CumulusMX/HttpStationEcowitt.cs @@ -1065,6 +1065,7 @@ public string ApplyData(string dataString, bool main, DateTime? ts = null) thisStation.UpdateStatusPanel(recDate); thisStation.UpdateMQTT(); + thisStation.LastDataReadTime = recDate; } catch (Exception ex) { diff --git a/CumulusMX/JsonStation.cs b/CumulusMX/JsonStation.cs index 5689646b..0c79668a 100644 --- a/CumulusMX/JsonStation.cs +++ b/CumulusMX/JsonStation.cs @@ -754,6 +754,7 @@ private string ApplyData(string dataString) UpdateStatusPanel(data.lastupdated); UpdateMQTT(); + LastDataReadTime = data.lastupdated; return retStr.ToString(); } diff --git a/CumulusMX/WeatherStation.cs b/CumulusMX/WeatherStation.cs index 86f0b72b..a5a4b639 100644 --- a/CumulusMX/WeatherStation.cs +++ b/CumulusMX/WeatherStation.cs @@ -325,7 +325,7 @@ public struct DailyHighLow public double[] WMR200ExtraTempValues { get; set; } - public DateTime lastDataReadTime; + public DateTime LastDataReadTime; public bool haveReadData = false; public bool ExtraSensorsDetected = false; From 61f61207dc98ea6f8528bb61b945bfbc954217b4 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Tue, 17 Sep 2024 22:02:42 +0100 Subject: [PATCH 43/63] First cut of New day file query web tag <#DayFileQuery> --- CumulusMX/CumulusMX.csproj | 4 +- CumulusMX/QueryDayFile.cs | 200 ++++++++++++++++++++++++++++++++++++ CumulusMX/WeatherStation.cs | 153 ++++++++++++++------------- CumulusMX/webtags.cs | 43 +++++++- Updates.txt | 5 +- 5 files changed, 329 insertions(+), 76 deletions(-) create mode 100644 CumulusMX/QueryDayFile.cs diff --git a/CumulusMX/CumulusMX.csproj b/CumulusMX/CumulusMX.csproj index 68f015ef..574e0a4b 100644 --- a/CumulusMX/CumulusMX.csproj +++ b/CumulusMX/CumulusMX.csproj @@ -95,12 +95,12 @@ - + - + diff --git a/CumulusMX/QueryDayFile.cs b/CumulusMX/QueryDayFile.cs new file mode 100644 index 00000000..798c47d6 --- /dev/null +++ b/CumulusMX/QueryDayFile.cs @@ -0,0 +1,200 @@ +using System; +using System.Globalization; +using System.Linq; + +using SQLite; + + +namespace CumulusMX +{ + + internal class QueryDayFile(SQLiteConnection databaseConnection) + { + private readonly SQLiteConnection db = databaseConnection; + private readonly static CultureInfo inv = CultureInfo.InvariantCulture; + internal static readonly string[] funcNameArray = { "min", "max", "sum", "avg", "count" }; + + + public (double value, DateTime time) DayFile(string propertyName, string function, string where, string from, string to) + { + var fromDate = DateTime.MinValue; + var toDate = DateTime.MinValue; + var byMonth = string.Empty; + + if (!funcNameArray.Contains(function)) + { + throw new ArgumentException($"Invalid function name - '{to}'"); + } + + + try + { + switch (from) + { + case "ThisYear": + fromDate = new DateTime(DateTime.Now.Year, 1, 1, 0, 0, 0, DateTimeKind.Local); + toDate = DateTime.Now; + break; + + case "ThisMonth": + fromDate = new DateTime(DateTime.Now.Year, DateTime.Now.Month, 1, 0, 0, 0, DateTimeKind.Local); + toDate = fromDate.AddMonths(1); + break; + + case string s when s.StartsWith("Month-"): + var rel = int.Parse(s.Split('-')[1]); + fromDate = new DateTime(DateTime.Now.Year, DateTime.Now.Month, 1, 0, 0, 0, DateTimeKind.Local).AddMonths(-rel); + toDate = fromDate.AddMonths(1); + break; + + case string s when s.StartsWith("Year-"): + rel = int.Parse(s.Split('-')[1]); + fromDate = new DateTime(DateTime.Now.Year, 1, 1, 0, 0, 0, DateTimeKind.Local).AddYears(-rel); + toDate = fromDate.AddYears(1); + break; + + case string s when s.Length > 5 && s.StartsWith("Month"): + // by month + byMonth = ("0" + s[5..])[^2..]; + + if (int.Parse(byMonth) > 12 || int.Parse(byMonth) < 1) + { + throw new ArgumentException(s + " exceeds month range 1-12!"); + } + break; + + + default: + // assume a date range + if (DateTime.TryParseExact(from, "yyyy-MM-dd", inv, DateTimeStyles.AssumeLocal, out fromDate)) + { + if (!DateTime.TryParseExact(to, "yyyy-MM-dd", inv, DateTimeStyles.AssumeLocal, out toDate)) + { + throw new ArgumentException($"Invalid toDate format - '{to}'"); + } + } + else + { + throw new ArgumentException($"Invalid fromDate format - '{from}'"); + } + break; + } + } + catch (Exception ex) + { + throw new ArgumentException("Error parsing from/to dates: " + ex.Message); + } + + DateTime logTime = DateTime.MinValue; + double value = -9999; + + var timeProp = propToTime(propertyName); + + if (function == "count") + { + if (byMonth == string.Empty) + { + value = db.ExecuteScalar($"SELECT COUNT({propertyName}) FROM DayFileRec WHERE {propertyName} {where} AND Date >= ? AND Date < ?", fromDate, toDate); + } + else + { + var ret = db.Query($"SELECT value, year_month FROM (SELECT strftime('%Y-%m', Date) AS year_month, COUNT(*) AS value FROM DayFileRec WHERE {propertyName} {where} AND strftime('%m', Date) = '{byMonth}' GROUP BY year_month) AS grouped_data ORDER BY value DESC LIMIT 1"); + if (ret.Count == 1) + { + value = ret[0].value; + var arr = ret[0].year_month.Split("-"); + logTime = new DateTime(int.Parse(arr[0]), int.Parse(arr[1]), 1, 0, 0, 0, DateTimeKind.Local); + } + } + } + else + { + if (byMonth == string.Empty) + { + var ret = db.Query($"SELECT {propertyName} value, {timeProp} time FROM DayFileRec WHERE {propertyName} = (SELECT {function}({propertyName}) FROM DayFileRec WHERE Date >= ? AND Date < ?) AND Date >= ? AND Date < ?", fromDate, toDate, fromDate, toDate); + if (ret.Count == 1) + { + value = ret[0].value; + logTime = ret[0].time; + } + } + else + { + //value = db.ExecuteScalar($"SELECT {function}({propertyName}) FROM DayFileRec WHERE strftime('%m', Date) = '?' ORDER BY Date", byMonth); + var sort = function == "min" ? "ASC" : "DESC"; + //var ret = db.Query($"SELECT {propertyName} value, {timeProp} time FROM DayFileRec WHERE {propertyName} = (SELECT {function}({propertyName}) FROM DayFileRec WHERE strftime('%m', Date) = '{byMonth}') AS grouped_data ORDER BY Date LIMIT 1"); + + var ret = db.Query($"SELECT {propertyName} value, {timeProp} time FROM DayFileRec WHERE {propertyName} = (SELECT {function}({propertyName}) FROM DayFileRec WHERE strftime('%m', Date) = '{byMonth}') AND strftime('%m', Date) = '{byMonth}'"); + + if (ret.Count == 1) + { + value = ret[0].value; + logTime = ret[0].time; + } + } + } + + return (value, logTime); + } + + private static string propToTime(string prop) + { + return prop switch + { + "HighGust" => "HighGustTime", + "HighGustBearing" => "Date", + "WindRun" => "Date", + "HighAvgWind" => "HighAvgWindTime", + "DominantWindBearing" => "Date", + + "LowTemp" => "LowTempTime", + "HighTemp" => "HighTempTime", + "AvgTemp" => "Date", + "HighHeatIndex" => "HighHeatIndexTime", + "HighAppTemp" => "HighAppTempTime", + "LowAppTemp" => "LowAppTempTime", + "LowWindChill" => "LowWindChillTime", + "HighDewPoint" => "HighDewPointTime", + "LowDewPoint" => "LowDewPoint", + "HighFeelsLike" => "HighFeelsLikeTime", + "LowFeelsLike" => "LowFeelsLikeTime", + "HighHumidex" => "HighHumidexTime", + + "LowPress" => "LowPressTime", + "HighPress" => "HighPressTime", + "HighRainRate" => "HighRainRateTime", + "TotalRain" => "Date", + "HighHourlyRain" => "HighHourlyRainTime", + "HighRain24h" => "HighRain24hTime", + + "LowHumidity" => "LowHumidityTime", + "HighHumidity" => "HighHumidityTime", + + "SunShineHours" => "Date", + "HighSolar" => "HighSolarTime", + "HighUv" => "HighUvTime", + "ET" => "Date", + + "HeatingDegreeDays" => "Date", + "CoolingDegreeDays" => "Date", + "ChillHours" => "Date", + + _ => "Invalid" + }; + } + + private class retValTime + { + public double value { get; set; } + public DateTime time { get; set; } + } + + private class retValString + { + public int value { get; set; } + public string year_month { get; set; } + } + + } +} + diff --git a/CumulusMX/WeatherStation.cs b/CumulusMX/WeatherStation.cs index a5a4b639..fd21f26d 100644 --- a/CumulusMX/WeatherStation.cs +++ b/CumulusMX/WeatherStation.cs @@ -104,63 +104,64 @@ public struct LogFileRec public double Humidex; } - public struct DayFileRec - { - public DateTime Date; - public double HighGust; - public int HighGustBearing; - public DateTime HighGustTime; - public double LowTemp; - public DateTime LowTempTime; - public double HighTemp; - public DateTime HighTempTime; - public double LowPress; - public DateTime LowPressTime; - public double HighPress; - public DateTime HighPressTime; - public double HighRainRate; - public DateTime HighRainRateTime; - public double TotalRain; - public double AvgTemp; - public double WindRun; - public double HighAvgWind; - public DateTime HighAvgWindTime; - public int LowHumidity; - public DateTime LowHumidityTime; - public int HighHumidity; - public DateTime HighHumidityTime; - public double ET; - public double SunShineHours; - public double HighHeatIndex; - public DateTime HighHeatIndexTime; - public double HighAppTemp; - public DateTime HighAppTempTime; - public double LowAppTemp; - public DateTime LowAppTempTime; - public double HighHourlyRain; - public DateTime HighHourlyRainTime; - public double LowWindChill; - public DateTime LowWindChillTime; - public double HighDewPoint; - public DateTime HighDewPointTime; - public double LowDewPoint; - public DateTime LowDewPointTime; - public int DominantWindBearing; - public double HeatingDegreeDays; - public double CoolingDegreeDays; - public int HighSolar; - public DateTime HighSolarTime; - public double HighUv; - public DateTime HighUvTime; - public double HighFeelsLike; - public DateTime HighFeelsLikeTime; - public double LowFeelsLike; - public DateTime LowFeelsLikeTime; - public double HighHumidex; - public DateTime HighHumidexTime; - public double ChillHours; - public double HighRain24h; - public DateTime HighRain24hTime; + public class DayFileRec + { + [PrimaryKey] + public DateTime Date { get; set; } + public double HighGust { get; set; } + public int HighGustBearing { get; set; } + public DateTime HighGustTime { get; set; } + public double LowTemp { get; set; } + public DateTime LowTempTime { get; set; } + public double HighTemp { get; set; } + public DateTime HighTempTime { get; set; } + public double LowPress { get; set; } + public DateTime LowPressTime { get; set; } + public double HighPress { get; set; } + public DateTime HighPressTime { get; set; } + public double HighRainRate { get; set; } + public DateTime HighRainRateTime { get; set; } + public double TotalRain { get; set; } + public double AvgTemp { get; set; } + public double WindRun { get; set; } + public double HighAvgWind { get; set; } + public DateTime HighAvgWindTime { get; set; } + public int LowHumidity { get; set; } + public DateTime LowHumidityTime { get; set; } + public int HighHumidity { get; set; } + public DateTime HighHumidityTime { get; set; } + public double ET { get; set; } + public double SunShineHours { get; set; } + public double HighHeatIndex { get; set; } + public DateTime HighHeatIndexTime { get; set; } + public double HighAppTemp { get; set; } + public DateTime HighAppTempTime { get; set; } + public double LowAppTemp { get; set; } + public DateTime LowAppTempTime { get; set; } + public double HighHourlyRain { get; set; } + public DateTime HighHourlyRainTime { get; set; } + public double LowWindChill { get; set; } + public DateTime LowWindChillTime { get; set; } + public double HighDewPoint { get; set; } + public DateTime HighDewPointTime { get; set; } + public double LowDewPoint { get; set; } + public DateTime LowDewPointTime { get; set; } + public int DominantWindBearing { get; set; } + public double HeatingDegreeDays { get; set; } + public double CoolingDegreeDays { get; set; } + public int HighSolar { get; set; } + public DateTime HighSolarTime { get; set; } + public double HighUv { get; set; } + public DateTime HighUvTime { get; set; } + public double HighFeelsLike { get; set; } + public DateTime HighFeelsLikeTime { get; set; } + public double LowFeelsLike { get; set; } + public DateTime LowFeelsLikeTime { get; set; } + public double HighHumidex { get; set; } + public DateTime HighHumidexTime { get; set; } + public double ChillHours { get; set; } + public double HighRain24h { get; set; } + public DateTime HighRain24hTime { get; set; } } public List DayFile = []; @@ -375,6 +376,8 @@ public virtual string GetEcowittVideoUrl() public int ExtraStationFreeMemory; public int StationRuntime; + public QueryDayFile DayFileQuery; + protected WeatherStation(Cumulus cumulus, bool extraStation = false) { @@ -406,6 +409,22 @@ protected WeatherStation(Cumulus cumulus, bool extraStation = false) WindRecent = new TWindRecent[MaxWindRecent]; WindVec = new TWindVec[MaxWindRecent]; + // Open database (create file if it doesn't exist) + SQLiteOpenFlags flags = SQLiteOpenFlags.Create | SQLiteOpenFlags.ReadWrite; + + RecentDataDb = new SQLiteConnection(new SQLiteConnectionString(cumulus.dbfile, flags, false, null, null, null, null, "yyyy-MM-dd HH:mm:ss")); + CheckSqliteDatabase(false); + RecentDataDb.CreateTable(); + RecentDataDb.CreateTable(); + RecentDataDb.CreateTable(); + RecentDataDb.Execute("create table if not exists WindRecentPointer (pntr INTEGER)"); + RecentDataDb.CreateTable(); + // switch off full synchronisation - the data base isn't that critical and we get a performance boost + RecentDataDb.Execute("PRAGMA synchronous = NORMAL"); + + // preload the failed sql cache - if any + ReloadFailedMySQLCommands(); + ReadTodayFile(); ReadYesterdayFile(); ReadAlltimeIniFile(); @@ -426,24 +445,11 @@ protected WeatherStation(Cumulus cumulus, bool extraStation = false) GetRainCounter(); GetRainFallTotals(); - // Open database (create file if it doesn't exist) - SQLiteOpenFlags flags = SQLiteOpenFlags.Create | SQLiteOpenFlags.ReadWrite; - - RecentDataDb = new SQLiteConnection(new SQLiteConnectionString(cumulus.dbfile, flags, false, null, null, null, null, "yyyy-MM-dd HH:mm:ss")); - CheckSqliteDatabase(false); - RecentDataDb.CreateTable(); - RecentDataDb.CreateTable(); - RecentDataDb.CreateTable(); - RecentDataDb.Execute("create table if not exists WindRecentPointer (pntr INTEGER)"); - // switch off full synchronisation - the data base isn't that critical and we get a performance boost - RecentDataDb.Execute("PRAGMA synchronous = NORMAL"); - - // preload the failed sql cache - if any - ReloadFailedMySQLCommands(); - versionCheckTime = new DateTime(1, 1, 1, Program.RandGenerator.Next(0, 23), Program.RandGenerator.Next(0, 59), 0, DateTimeKind.Local); SensorReception = []; + + DayFileQuery = new QueryDayFile(RecentDataDb); } private void CheckSqliteDatabase(bool giveup) @@ -8943,6 +8949,7 @@ public string LoadDayFile() // Clear the existing list DayFile.Clear(); + RecentDataDb.Execute("DELETE FROM DayFileRec"); var lines = File.ReadAllLines(cumulus.DayFileName); @@ -8978,6 +8985,8 @@ public string LoadDayFile() } } + RecentDataDb.InsertAll(DayFile); + watch.Stop(); cumulus.LogDebugMessage($"LoadDayFile: Dayfile parse = {watch.ElapsedMilliseconds} ms"); diff --git a/CumulusMX/webtags.cs b/CumulusMX/webtags.cs index 9db65a31..3aa43a0b 100644 --- a/CumulusMX/webtags.cs +++ b/CumulusMX/webtags.cs @@ -5928,6 +5928,45 @@ private string TagMySqlIntervalTime(Dictionary tagParams) return GetFormattedDateTime(cumulus.MySqlILastntervalTime, "yyyy-MM-dd HH:mm", tagParams); } + private string TagQueryDayFile(Dictionary tagParams) + { + var value = tagParams.Get("value"); + var function = tagParams.Get("function"); + var where = tagParams.Get("where"); + var from = tagParams.Get("from"); + var to = tagParams.Get("to"); + var showDate = tagParams.Get("showDate"); + + tagParams.Add("tc", function == "count" ? "y" : "n"); + + var defaultFormat = function == "count" ? "MM/yyyy" : "g"; + + var ret = station.DayFileQuery.DayFile(value, function, where, from, to); + + if (ret.value < -9998) + { + if (showDate == "y") + { + return "[\"-\",\"-\"]"; + } + else + { + return "-"; + } + } + else + { + if (showDate == "y") + { + return "[\"" + CheckRcDp(ret.value, tagParams, 1) + "\",\"" + GetFormattedDateTime(ret.time, defaultFormat, tagParams) + "\"]"; + } + else + { + return CheckRcDp(ret.value, tagParams, 1); + } + } + } + public void InitialiseWebtags() { // create the web tag dictionary @@ -6868,7 +6907,9 @@ public void InitialiseWebtags() { "Option_showUV", TagOption_showUV }, // MySQL insert times { "MySqlRealtimeTime", TagMySqlRealtimeTime }, - { "MySqlIntervalTime", TagMySqlIntervalTime } + { "MySqlIntervalTime", TagMySqlIntervalTime }, + // General queries + { "QueryDayFile", TagQueryDayFile } }; cumulus.LogMessage(webTagDictionary.Count + " web tags initialised"); diff --git a/Updates.txt b/Updates.txt index 60d0a49a..533016fe 100644 --- a/Updates.txt +++ b/Updates.txt @@ -7,7 +7,8 @@ BETA 4036 Changes: - Add Soil moisture sensors 9-16 to Ecowitt TCP API decode of sensor info - Soil moisture units now follow the source - Two new web tags <#NewRecordAlarm> and <#NewRecordAlarmMessage> -- Cleanup old MD5 hash files +- Delete old MD5 hash files +- New day file query web tag <#DayFileQuery>. Please read the separate documention (QueryDayFile.md) for more details BETA 4035 Changes: @@ -63,6 +64,8 @@ New - NewRecordAlarm somewhat replicates the existing #newrecord web tag, but is also controlled by the alarm being enable/disabled - NewRecordAlarmMessage displays the last new record alarm text message - Old MD5 hash files are now deleted on startup +- New web tag <#DayFileQuery> which allows flexible querying of the day file. Please read the separate documention (QueryDayFile.md) for more details + Changed - Davis WLL/Davis Cloud stations now use the WL.com subscription level to determine if they use WL.com as a logger From 12e688b076211f2413e9e21651386ed97b63dd64 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Tue, 17 Sep 2024 22:05:35 +0100 Subject: [PATCH 44/63] tidying --- CumulusMX/QueryDayFile.cs | 7 +------ 1 file changed, 1 insertion(+), 6 deletions(-) diff --git a/CumulusMX/QueryDayFile.cs b/CumulusMX/QueryDayFile.cs index 798c47d6..975fc5e1 100644 --- a/CumulusMX/QueryDayFile.cs +++ b/CumulusMX/QueryDayFile.cs @@ -120,10 +120,6 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) } else { - //value = db.ExecuteScalar($"SELECT {function}({propertyName}) FROM DayFileRec WHERE strftime('%m', Date) = '?' ORDER BY Date", byMonth); - var sort = function == "min" ? "ASC" : "DESC"; - //var ret = db.Query($"SELECT {propertyName} value, {timeProp} time FROM DayFileRec WHERE {propertyName} = (SELECT {function}({propertyName}) FROM DayFileRec WHERE strftime('%m', Date) = '{byMonth}') AS grouped_data ORDER BY Date LIMIT 1"); - var ret = db.Query($"SELECT {propertyName} value, {timeProp} time FROM DayFileRec WHERE {propertyName} = (SELECT {function}({propertyName}) FROM DayFileRec WHERE strftime('%m', Date) = '{byMonth}') AND strftime('%m', Date) = '{byMonth}'"); if (ret.Count == 1) @@ -142,10 +138,8 @@ private static string propToTime(string prop) return prop switch { "HighGust" => "HighGustTime", - "HighGustBearing" => "Date", "WindRun" => "Date", "HighAvgWind" => "HighAvgWindTime", - "DominantWindBearing" => "Date", "LowTemp" => "LowTempTime", "HighTemp" => "HighTempTime", @@ -162,6 +156,7 @@ private static string propToTime(string prop) "LowPress" => "LowPressTime", "HighPress" => "HighPressTime", + "HighRainRate" => "HighRainRateTime", "TotalRain" => "Date", "HighHourlyRain" => "HighHourlyRainTime", From 53b1c43a0c9a3c3b580400e05243180feeea2156 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Tue, 17 Sep 2024 22:07:54 +0100 Subject: [PATCH 45/63] More dayfilequery file tidy --- CumulusMX/QueryDayFile.cs | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/CumulusMX/QueryDayFile.cs b/CumulusMX/QueryDayFile.cs index 975fc5e1..7a16c0a4 100644 --- a/CumulusMX/QueryDayFile.cs +++ b/CumulusMX/QueryDayFile.cs @@ -98,7 +98,7 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) } else { - var ret = db.Query($"SELECT value, year_month FROM (SELECT strftime('%Y-%m', Date) AS year_month, COUNT(*) AS value FROM DayFileRec WHERE {propertyName} {where} AND strftime('%m', Date) = '{byMonth}' GROUP BY year_month) AS grouped_data ORDER BY value DESC LIMIT 1"); + var ret = db.Query($"SELECT value, year_month FROM (SELECT strftime('%Y-%m', Date) AS year_month, COUNT(*) AS value FROM DayFileRec WHERE {propertyName} {where} AND strftime('%m', Date) = '{byMonth}' GROUP BY year_month) AS grouped_data ORDER BY value DESC LIMIT 1"); if (ret.Count == 1) { value = ret[0].value; @@ -111,7 +111,7 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) { if (byMonth == string.Empty) { - var ret = db.Query($"SELECT {propertyName} value, {timeProp} time FROM DayFileRec WHERE {propertyName} = (SELECT {function}({propertyName}) FROM DayFileRec WHERE Date >= ? AND Date < ?) AND Date >= ? AND Date < ?", fromDate, toDate, fromDate, toDate); + var ret = db.Query($"SELECT {propertyName} value, {timeProp} time FROM DayFileRec WHERE {propertyName} = (SELECT {function}({propertyName}) FROM DayFileRec WHERE Date >= ? AND Date < ?) AND Date >= ? AND Date < ?", fromDate, toDate, fromDate, toDate); if (ret.Count == 1) { value = ret[0].value; @@ -120,7 +120,7 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) } else { - var ret = db.Query($"SELECT {propertyName} value, {timeProp} time FROM DayFileRec WHERE {propertyName} = (SELECT {function}({propertyName}) FROM DayFileRec WHERE strftime('%m', Date) = '{byMonth}') AND strftime('%m', Date) = '{byMonth}'"); + var ret = db.Query($"SELECT {propertyName} value, {timeProp} time FROM DayFileRec WHERE {propertyName} = (SELECT {function}({propertyName}) FROM DayFileRec WHERE strftime('%m', Date) = '{byMonth}') AND strftime('%m', Date) = '{byMonth}'"); if (ret.Count == 1) { @@ -178,13 +178,13 @@ private static string propToTime(string prop) }; } - private class retValTime + private sealed class RetValTime { public double value { get; set; } public DateTime time { get; set; } } - private class retValString + private sealed class RetValString { public int value { get; set; } public string year_month { get; set; } From d869856160fc63eadda4ef1137aa98594efd856d Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Wed, 18 Sep 2024 22:45:06 +0100 Subject: [PATCH 46/63] Yet more work on the Query Day File stuff --- CumulusMX/Api.cs | 33 ++++++++ CumulusMX/QueryDayFile.cs | 160 ++++++++++++++++++++++++++++++++--- CumulusMX/StationSettings.cs | 2 +- CumulusMX/webtags.cs | 15 ++-- Updates.txt | 4 +- 5 files changed, 191 insertions(+), 23 deletions(-) diff --git a/CumulusMX/Api.cs b/CumulusMX/Api.cs index c90063cf..3685beb4 100644 --- a/CumulusMX/Api.cs +++ b/CumulusMX/Api.cs @@ -9,6 +9,10 @@ using EmbedIO.Routing; using EmbedIO.WebApi; +using Org.BouncyCastle.Ocsp; + +using ServiceStack.Text.Common; + namespace CumulusMX { @@ -934,6 +938,35 @@ public async Task GetThisPeriodRecordData() } } + [Route(HttpVerbs.Post, "/records/query/{req}")] + public async Task GetQueryData(string req) + { + Response.ContentType = "application/json"; + + using var writer = HttpContext.OpenResponseText(new UTF8Encoding(false)); + + if (Station == null) + { + await writer.WriteAsync("{}"); + return; + } + + try + { + switch (req) + { + case "dayfile.json": + await writer.WriteAsync(Station.DayFileQuery.WebQuery(HttpContext)); + break; + } + } + catch (Exception ex) + { + cumulus.LogErrorMessage($"api/query/{req}: Unexpected Error, Description: \"{ex.Message}\""); + Response.StatusCode = 500; + } + } + private static string EscapeUnicode(string input) { StringBuilder sb = new StringBuilder(input.Length); diff --git a/CumulusMX/QueryDayFile.cs b/CumulusMX/QueryDayFile.cs index 7a16c0a4..aef66128 100644 --- a/CumulusMX/QueryDayFile.cs +++ b/CumulusMX/QueryDayFile.cs @@ -1,6 +1,13 @@ using System; +using System.Collections.Generic; using System.Globalization; +using System.IO; using System.Linq; +using System.Net; + +using EmbedIO; + +using ServiceStack.Text; using SQLite; @@ -15,11 +22,12 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) internal static readonly string[] funcNameArray = { "min", "max", "sum", "avg", "count" }; - public (double value, DateTime time) DayFile(string propertyName, string function, string where, string from, string to) + public (double value, DateTime time) DayFile(string propertyName, string function, string where, string from, string to, string resfunc) { var fromDate = DateTime.MinValue; var toDate = DateTime.MinValue; var byMonth = string.Empty; + var yearly = false; if (!funcNameArray.Contains(function)) { @@ -63,6 +71,9 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) } break; + case "Yearly": + yearly = true; + break; default: // assume a date range @@ -88,30 +99,63 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) DateTime logTime = DateTime.MinValue; double value = -9999; - var timeProp = propToTime(propertyName); + var timeProp = function == "sum" ? "Date" : propToTime(propertyName); + + // Combinations:- + // count: + // where [mandatory] + // date range + // monthly - this requires the addition of the count function to return the highest or lowest count if (function == "count") { - if (byMonth == string.Empty) + if (byMonth == string.Empty && !yearly) { value = db.ExecuteScalar($"SELECT COUNT({propertyName}) FROM DayFileRec WHERE {propertyName} {where} AND Date >= ? AND Date < ?", fromDate, toDate); + logTime = fromDate; } else { - var ret = db.Query($"SELECT value, year_month FROM (SELECT strftime('%Y-%m', Date) AS year_month, COUNT(*) AS value FROM DayFileRec WHERE {propertyName} {where} AND strftime('%m', Date) = '{byMonth}' GROUP BY year_month) AS grouped_data ORDER BY value DESC LIMIT 1"); - if (ret.Count == 1) + var sort = resfunc == "max" ? "DESC" : "ASC"; + + if (yearly) { - value = ret[0].value; - var arr = ret[0].year_month.Split("-"); - logTime = new DateTime(int.Parse(arr[0]), int.Parse(arr[1]), 1, 0, 0, 0, DateTimeKind.Local); + var ret = db.Query($"SELECT value, year FROM (SELECT strftime('%Y', Date) AS year, COUNT(*) AS value FROM DayFileRec WHERE {propertyName} {where} GROUP BY year) AS grouped_data ORDER BY value {sort} LIMIT 1"); + + if (ret.Count == 1) + { + value = ret[0].value; + logTime = new DateTime(int.Parse(ret[0].year_month), 1, 1, 0, 0, 0, DateTimeKind.Local); + } + } + else + { + var ret = db.Query($"SELECT value, year_month FROM (SELECT strftime('%Y-%m', Date) AS year_month, COUNT(*) AS value FROM DayFileRec WHERE {propertyName} {where} AND strftime('%m', Date) = '{byMonth}' GROUP BY year_month) AS grouped_data ORDER BY value {sort} LIMIT 1"); + + if (ret.Count == 1) + { + value = ret[0].value; + var arr = ret[0].year_month.Split("-"); + logTime = new DateTime(int.Parse(arr[0]), int.Parse(arr[1]), 1, 0, 0, 0, DateTimeKind.Local); + } } } } else { - if (byMonth == string.Empty) + if (byMonth == string.Empty && !yearly) { - var ret = db.Query($"SELECT {propertyName} value, {timeProp} time FROM DayFileRec WHERE {propertyName} = (SELECT {function}({propertyName}) FROM DayFileRec WHERE Date >= ? AND Date < ?) AND Date >= ? AND Date < ?", fromDate, toDate, fromDate, toDate); + List ret; + + if (function == "avg" || function == "sum") + { + ret = db.Query($"SELECT {function}({propertyName}) value, {timeProp} time FROM DayFileRec WHERE Date >= ? AND Date < ?", fromDate, toDate); + } + else + { + ret = db.Query($"SELECT {propertyName} value, {timeProp} time FROM DayFileRec WHERE {propertyName} = (SELECT {function}({propertyName}) FROM DayFileRec WHERE Date >= ? AND Date < ?) AND Date >= ? AND Date < ? LIMIT 1", fromDate, toDate, fromDate, toDate); + } + if (ret.Count == 1) { value = ret[0].value; @@ -120,12 +164,41 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) } else { - var ret = db.Query($"SELECT {propertyName} value, {timeProp} time FROM DayFileRec WHERE {propertyName} = (SELECT {function}({propertyName}) FROM DayFileRec WHERE strftime('%m', Date) = '{byMonth}') AND strftime('%m', Date) = '{byMonth}'"); + var sort = resfunc == "max" ? "DESC" : "ASC"; - if (ret.Count == 1) + if (yearly) { - value = ret[0].value; - logTime = ret[0].time; + var ret = db.Query($"SELECT {function}({propertyName}) value, {timeProp} time, strftime('%Y', Date) year FROM DayFileRec GROUP BY year ORDER BY value {sort} LIMIT 1"); + + if (ret.Count == 1) + { + value = ret[0].value; + logTime = ret[0].time; + } + } + else + { + if (function == "sum" || function == "avg") + { + var ret = db.Query($"SELECT {function}({propertyName}) value, strftime('%Y-%m', Date) year_month FROM DayFileRec WHERE strftime('%m', Date) = '{byMonth}' GROUP BY year_month ORDER BY value {sort} LIMIT 1"); + + if (ret.Count == 1) + { + value = ret[0].value; + var arr = ret[0].year_month.Split("-"); + logTime = new DateTime(int.Parse(arr[0]), int.Parse(arr[1]), 1, 0, 0, 0, DateTimeKind.Local); + } + } + else + { + var ret = db.Query($"SELECT {function}({propertyName}) value, {timeProp} time FROM DayFileRec WHERE strftime('%m', Date) = '{byMonth}' GROUP BY strftime('%Y-%m', Date) ORDER BY value {sort} LIMIT 1"); + + if (ret.Count == 1) + { + value = ret[0].value; + logTime = ret[0].time; + } + } } } } @@ -133,6 +206,55 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) return (value, logTime); } + + public string WebQuery(IHttpContext context) + { + var errorMsg = string.Empty; + context.Response.StatusCode = 200; + // get the response + try + { + Program.cumulus.LogMessage("API Querying day file..."); + + var data = new StreamReader(context.Request.InputStream).ReadToEnd(); + + // Start at char 5 to skip the "json=" prefix + var json = WebUtility.UrlDecode(data)[5..]; + + // de-serialize it + var req = JsonSerializer.DeserializeFromString(json); + + // process the settings + try + { + var from = req.startsel == "User" ? req.start : req.startsel; + var format = "g"; + + var (value, time) = DayFile(req.dataname, req.function, req.where, from, req.end, req.countfunction); + + Program.cumulus.LogMessage("API Querying day file complete"); + + return $"{{\"value\": {value:0.000}, \"time\":\"{time.ToString(format)}\"}}"; + } + catch (Exception ex) + { + var msg = "Error day file query Options: " + ex.Message; + Program.cumulus.LogErrorMessage(msg); + errorMsg += msg + "\n\n"; + context.Response.StatusCode = 500; + } + } + catch (Exception ex) + { + Program.cumulus.LogErrorMessage("Query day file error: " + ex.Message); + context.Response.StatusCode = 500; + return ex.Message; + } + + return context.Response.StatusCode == 200 ? "success" : errorMsg; + } + + private static string propToTime(string prop) { return prop switch @@ -190,6 +312,16 @@ private sealed class RetValString public string year_month { get; set; } } + private sealed class WebReq + { + public string dataname { get; set; } + public string function { get; set; } + public string where { get; set; } + public string startsel { get; set; } + public string start { get; set; } + public string end { get; set; } + public string countfunction { get; set; } + } } } diff --git a/CumulusMX/StationSettings.cs b/CumulusMX/StationSettings.cs index 59bbecd9..527ecec7 100644 --- a/CumulusMX/StationSettings.cs +++ b/CumulusMX/StationSettings.cs @@ -1605,7 +1605,7 @@ internal string SetSelectaPeriodOptions(IHttpContext context) } catch (Exception ex) { - cumulus.LogErrorMessage("Update selecaperiod options error: " + ex.Message); + cumulus.LogErrorMessage("Update select-a-period options error: " + ex.Message); context.Response.StatusCode = 500; return ex.Message; } diff --git a/CumulusMX/webtags.cs b/CumulusMX/webtags.cs index 3aa43a0b..8f552b46 100644 --- a/CumulusMX/webtags.cs +++ b/CumulusMX/webtags.cs @@ -5936,12 +5936,13 @@ private string TagQueryDayFile(Dictionary tagParams) var from = tagParams.Get("from"); var to = tagParams.Get("to"); var showDate = tagParams.Get("showDate"); + var resfunc = tagParams.Get("resFunc"); tagParams.Add("tc", function == "count" ? "y" : "n"); var defaultFormat = function == "count" ? "MM/yyyy" : "g"; - var ret = station.DayFileQuery.DayFile(value, function, where, from, to); + var ret = station.DayFileQuery.DayFile(value, function, where, from, to, resfunc); if (ret.value < -9998) { @@ -6602,7 +6603,7 @@ public void InitialiseWebtags() { "AirLinkPct_24hrOut", AirLinkPct_24hrOut }, { "AirLinkPct_NowcastOut", AirLinkPct_NowcastOut }, - // Monthly highs and lows - values + // This month's highs and lows - values { "MonthTempH", TagMonthTempH }, { "MonthTempL", TagMonthTempL }, { "MonthHeatIndexH", TagMonthHeatIndexH }, @@ -6631,7 +6632,7 @@ public void InitialiseWebtags() { "MonthLongestWetPeriod", TagMonthLongestWetPeriod }, { "MonthHighDailyTempRange", TagMonthHighDailyTempRange }, { "MonthLowDailyTempRange", TagMonthLowDailyTempRange }, - // This year"s highs and lows - times + // This month's highs and lows - times { "MonthTempHT", TagMonthTempHt }, { "MonthTempLT", TagMonthTempLt }, { "MonthHeatIndexHT", TagMonthHeatIndexHt }, @@ -6652,7 +6653,7 @@ public void InitialiseWebtags() { "MonthRain24HourHT", TagMonthRain24HourHt }, { "MonthDewPointHT", TagMonthDewPointHt }, { "MonthDewPointLT", TagMonthDewPointLt }, - // This month"s highs and lows - dates + // This month's highs and lows - dates { "MonthTempHD", TagMonthTempHd }, { "MonthTempLD", TagMonthTempLd }, { "MonthHeatIndexHD", TagMonthHeatIndexHd }, @@ -6681,7 +6682,7 @@ public void InitialiseWebtags() { "MonthLongestWetPeriodD", TagMonthLongestWetPeriodD }, { "MonthHighDailyTempRangeD", TagMonthHighDailyTempRangeD }, { "MonthLowDailyTempRangeD", TagMonthLowDailyTempRangeD }, - // This Year"s highs and lows - values + // This Year's highs and lows - values { "YearTempH", TagYearTempH }, { "YearTempL", TagYearTempL }, { "YearHeatIndexH", TagYearHeatIndexH }, @@ -6711,7 +6712,7 @@ public void InitialiseWebtags() { "YearLongestWetPeriod", TagYearLongestWetPeriod }, { "YearHighDailyTempRange", TagYearHighDailyTempRange }, { "YearLowDailyTempRange", TagYearLowDailyTempRange }, - // This years"s highs and lows - times + // This years highs and lows - times { "YearTempHT", TagYearTempHt }, { "YearTempLT", TagYearTempLt }, { "YearHeatIndexHT", TagYearHeatIndexHt }, @@ -6732,7 +6733,7 @@ public void InitialiseWebtags() { "YearRain24HourHT", TagYearRain24HourHt }, { "YearDewPointHT", TagYearDewPointHt }, { "YearDewPointLT", TagYearDewPointLt }, - // Yearly highs and lows - dates + // This years highs and lows - dates { "YearTempHD", TagYearTempHd }, { "YearTempLD", TagYearTempLd }, { "YearHeatIndexHD", TagYearHeatIndexHd }, diff --git a/Updates.txt b/Updates.txt index 533016fe..7bcb5789 100644 --- a/Updates.txt +++ b/Updates.txt @@ -9,6 +9,8 @@ BETA 4036 Changes: - Two new web tags <#NewRecordAlarm> and <#NewRecordAlarmMessage> - Delete old MD5 hash files - New day file query web tag <#DayFileQuery>. Please read the separate documention (QueryDayFile.md) for more details +- Fix Ecowitt HTTP API station not showing in the config Wizard +- Add bash script for old Fine Offset USB stations on Linux BETA 4035 Changes: @@ -65,7 +67,7 @@ New - NewRecordAlarmMessage displays the last new record alarm text message - Old MD5 hash files are now deleted on startup - New web tag <#DayFileQuery> which allows flexible querying of the day file. Please read the separate documention (QueryDayFile.md) for more details - +- Added a script to /MXutils/linux/Fix_FineOffset_USB.sh to fix Fine Offset USB stations Changed - Davis WLL/Davis Cloud stations now use the WL.com subscription level to determine if they use WL.com as a logger From 9ceb7ce8cff51d4ee44d6b5aa66a4696ce0b32bd Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Thu, 19 Sep 2024 14:47:37 +0100 Subject: [PATCH 47/63] minor text fix to query day file error message --- CumulusMX/EcowittLocalApi.cs | 9 ++------- CumulusMX/QueryDayFile.cs | 12 +++++++++++- 2 files changed, 13 insertions(+), 8 deletions(-) diff --git a/CumulusMX/EcowittLocalApi.cs b/CumulusMX/EcowittLocalApi.cs index 0dcf856d..ff7d56ef 100644 --- a/CumulusMX/EcowittLocalApi.cs +++ b/CumulusMX/EcowittLocalApi.cs @@ -8,15 +8,10 @@ namespace CumulusMX { - internal sealed class EcowittLocalApi : IDisposable + internal sealed class EcowittLocalApi(Cumulus cumul) : IDisposable { - private readonly Cumulus cumulus; + private readonly Cumulus cumulus = cumul; private static readonly NumberFormatInfo invNum = CultureInfo.InvariantCulture.NumberFormat; - public EcowittLocalApi(Cumulus cumul) - { - cumulus = cumul; - } - public LiveData GetLiveData(CancellationToken token) { diff --git a/CumulusMX/QueryDayFile.cs b/CumulusMX/QueryDayFile.cs index aef66128..df085bb2 100644 --- a/CumulusMX/QueryDayFile.cs +++ b/CumulusMX/QueryDayFile.cs @@ -31,7 +31,7 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) if (!funcNameArray.Contains(function)) { - throw new ArgumentException($"Invalid function name - '{to}'"); + throw new ArgumentException($"Invalid function name - '{function}'"); } @@ -234,6 +234,16 @@ public string WebQuery(IHttpContext context) Program.cumulus.LogMessage("API Querying day file complete"); + if (value < -9998) + { + return "{\"value\": \"n/a\", \"time\": \"n/a\"}"; + } + + if (time == DateTime.MinValue) + { + return $"{{\"value\": {value:0.000}, \"time\":\"n/a\"}}"; + } + return $"{{\"value\": {value:0.000}, \"time\":\"{time.ToString(format)}\"}}"; } catch (Exception ex) From 25bc7dbed1831565b90e58fbbddd018f67c2f121 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Thu, 19 Sep 2024 15:32:49 +0100 Subject: [PATCH 48/63] Change what is logged to Lastet Errors to be the innermost exception message rather that the topmost --- CumulusMX/Cumulus.cs | 2 +- CumulusMX/EcowittHttpApiStation.cs | 10 ++++++++-- Updates.txt | 2 +- 3 files changed, 10 insertions(+), 4 deletions(-) diff --git a/CumulusMX/Cumulus.cs b/CumulusMX/Cumulus.cs index 5b5e43a9..4d7cd68e 100644 --- a/CumulusMX/Cumulus.cs +++ b/CumulusMX/Cumulus.cs @@ -11726,7 +11726,7 @@ public void LogExceptionMessage(Exception ex, string message, bool logError = tr { _ = ErrorList.Dequeue(); } - ErrorList.Enqueue((DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss - ") + message + " - " + ex.Message)); + ErrorList.Enqueue((DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss - ") + message + " - " + ex.GetInnerMostException().Message)); LatestError = message + " - " + ex.Message; LatestErrorTS = DateTime.Now; } diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index bac91f1d..5f32e083 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -159,10 +159,16 @@ public override void Start() dataLastRead = DateTime.Now; // process the common_list sensors - ProcessCommonList(rawData.common_list, dataLastRead); + if (rawData.common_list != null) + { + ProcessCommonList(rawData.common_list, dataLastRead); + } // process base station values - ProcessWh25(rawData.wh25, dataLastRead); + if (rawData.wh25 != null) + { + ProcessWh25(rawData.wh25, dataLastRead); + } // process rain values if (cumulus.Gw1000PrimaryRainSensor == 0 && rawData.rain != null) diff --git a/Updates.txt b/Updates.txt index 7bcb5789..6b12806c 100644 --- a/Updates.txt +++ b/Updates.txt @@ -8,7 +8,7 @@ BETA 4036 Changes: - Soil moisture units now follow the source - Two new web tags <#NewRecordAlarm> and <#NewRecordAlarmMessage> - Delete old MD5 hash files -- New day file query web tag <#DayFileQuery>. Please read the separate documention (QueryDayFile.md) for more details +- New day file query web tag <#DayFileQuery>. Please read the separate documentation (QueryDayFileWebTag.md) for more details - Fix Ecowitt HTTP API station not showing in the config Wizard - Add bash script for old Fine Offset USB stations on Linux From 4397863b2c1b5364a7a6933295e9bc14435f1a95 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Fri, 20 Sep 2024 14:25:32 +0100 Subject: [PATCH 49/63] New daily data now added to SQLite at rollover Sqlite day file data updated when dayfile is edited Station settings now match Wiard for HTTP API station --- CumulusMX/DataEditor.cs | 5 ++++- CumulusMX/QueryDayFile.cs | 2 ++ CumulusMX/StationSettings.cs | 3 --- CumulusMX/WeatherStation.cs | 3 ++- 4 files changed, 8 insertions(+), 5 deletions(-) diff --git a/CumulusMX/DataEditor.cs b/CumulusMX/DataEditor.cs index d612246d..b8bb6639 100644 --- a/CumulusMX/DataEditor.cs +++ b/CumulusMX/DataEditor.cs @@ -3015,11 +3015,14 @@ internal string EditDayFile(IHttpContext context) { lines[lineNum] = newLine; - // Update the in memory record try { + // Update the in memory record station.DayFile[lineNum] = station.ParseDayFileRec(newLine); + // update SQLite + station.RecentDataDb.Update(station.DayFile[lineNum]); + // write dayfile back again File.WriteAllLines(cumulus.DayFileName, lines); cumulus.LogMessage($"EditDayFile: Changed dayfile line {lineNum + 1}, original = {orgLine}"); diff --git a/CumulusMX/QueryDayFile.cs b/CumulusMX/QueryDayFile.cs index df085bb2..a5ee3621 100644 --- a/CumulusMX/QueryDayFile.cs +++ b/CumulusMX/QueryDayFile.cs @@ -34,6 +34,7 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) throw new ArgumentException($"Invalid function name - '{function}'"); } + Program.cumulus.LogDebugMessage($"QueryDayFile: prop={propertyName}, func={function}, where={where}, from={from}, to={to}, resFunc={resfunc}"); try { @@ -202,6 +203,7 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) } } } + Program.cumulus.LogDebugMessage($"QueryDayFile: Result = [{value:0.00}, {logTime:g}]"); return (value, logTime); } diff --git a/CumulusMX/StationSettings.cs b/CumulusMX/StationSettings.cs index 527ecec7..f40b948e 100644 --- a/CumulusMX/StationSettings.cs +++ b/CumulusMX/StationSettings.cs @@ -161,7 +161,6 @@ internal string GetAlpacaFormData() var ecowittHttpApi = new JsonStationSettingsHttpApi() { ipaddress = cumulus.Gw1000IpAddress, - macaddress = cumulus.Gw1000MacAddress, password = cumulus.EcowittHttpPassword }; @@ -920,7 +919,6 @@ internal string UpdateConfig(IHttpContext context) if (settings.ecowitthttpapi != null) { cumulus.Gw1000IpAddress = string.IsNullOrWhiteSpace(settings.ecowitthttpapi.ipaddress) ? null : settings.ecowitthttpapi.ipaddress.Trim(); - cumulus.Gw1000MacAddress = string.IsNullOrWhiteSpace(settings.ecowitthttpapi.macaddress) ? null : settings.ecowitthttpapi.macaddress.Trim().ToUpper(); cumulus.EcowittHttpPassword = string.IsNullOrWhiteSpace(settings.ecowitthttpapi.password) ? null : settings.ecowitthttpapi.password.Trim(); } } @@ -1811,7 +1809,6 @@ internal class JsonStationSettingsGw1000Conn internal class JsonStationSettingsHttpApi { public string ipaddress { get; set; } - public string macaddress { get; set; } public string password { get; set; } } diff --git a/CumulusMX/WeatherStation.cs b/CumulusMX/WeatherStation.cs index fd21f26d..bf679dd4 100644 --- a/CumulusMX/WeatherStation.cs +++ b/CumulusMX/WeatherStation.cs @@ -7897,7 +7897,8 @@ private async Task DoDayfile(DateTime timestamp) DayFile.Add(newRec); - + // add to SQLite + RecentDataDb.Insert(newRec); if (cumulus.MySqlSettings.Dayfile.Enabled) { From 04df565b962e9967c8b4a2707db7d8d34a7ef09d Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Fri, 20 Sep 2024 14:50:11 +0100 Subject: [PATCH 50/63] Query Daily Data to date now inclusive --- CumulusMX/CumulusMX.csproj | 2 +- CumulusMX/QueryDayFile.cs | 6 +++--- Updates.txt | 10 +++++++++- 3 files changed, 13 insertions(+), 5 deletions(-) diff --git a/CumulusMX/CumulusMX.csproj b/CumulusMX/CumulusMX.csproj index 574e0a4b..c694d44e 100644 --- a/CumulusMX/CumulusMX.csproj +++ b/CumulusMX/CumulusMX.csproj @@ -55,7 +55,7 @@ - 4.2.0.4036-beta + 4.2.0.4037-beta Copyright © 2015-$([System.DateTime]::Now.ToString('yyyy')) Cumulus MX $(AssemblyName) diff --git a/CumulusMX/QueryDayFile.cs b/CumulusMX/QueryDayFile.cs index a5ee3621..f53f42b5 100644 --- a/CumulusMX/QueryDayFile.cs +++ b/CumulusMX/QueryDayFile.cs @@ -112,7 +112,7 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) { if (byMonth == string.Empty && !yearly) { - value = db.ExecuteScalar($"SELECT COUNT({propertyName}) FROM DayFileRec WHERE {propertyName} {where} AND Date >= ? AND Date < ?", fromDate, toDate); + value = db.ExecuteScalar($"SELECT COUNT({propertyName}) FROM DayFileRec WHERE {propertyName} {where} AND Date BETWEEN ? AND ?", fromDate, toDate); logTime = fromDate; } else @@ -150,11 +150,11 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) if (function == "avg" || function == "sum") { - ret = db.Query($"SELECT {function}({propertyName}) value, {timeProp} time FROM DayFileRec WHERE Date >= ? AND Date < ?", fromDate, toDate); + ret = db.Query($"SELECT {function}({propertyName}) value, {timeProp} time FROM DayFileRec WHERE Date BETWEEN ? AND ?", fromDate, toDate); } else { - ret = db.Query($"SELECT {propertyName} value, {timeProp} time FROM DayFileRec WHERE {propertyName} = (SELECT {function}({propertyName}) FROM DayFileRec WHERE Date >= ? AND Date < ?) AND Date >= ? AND Date < ? LIMIT 1", fromDate, toDate, fromDate, toDate); + ret = db.Query($"SELECT {propertyName} value, {timeProp} time FROM DayFileRec WHERE {propertyName} = (SELECT {function}({propertyName}) FROM DayFileRec WHERE Date BETWEEN ? AND ?) AND Date BETWEEN ? AND ? LIMIT 1", fromDate, toDate, fromDate, toDate); } if (ret.Count == 1) diff --git a/Updates.txt b/Updates.txt index 6b12806c..18c8ed31 100644 --- a/Updates.txt +++ b/Updates.txt @@ -1,5 +1,13 @@ -4.2.0 - b4036 +4.2.0 - b4037 ————————————— +BETA 4037 Changes: +—————————————————— +- Configuration Wizard now working again +- Station Settings for HTTP API station now drops MAC to match Wizard settings +- Query Daily Data to date now inclusive +- Daily data for query now added to at rollover, and amended at day file edits + + BETA 4036 Changes: —————————————————— - Accessibility updates for the Interval and Daily data viewer pages From 2e8a45388b744fb5cd41098b1e3ebee1f62d6483 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Fri, 20 Sep 2024 15:03:04 +0100 Subject: [PATCH 51/63] Ooops, for to make ThisMonth etc match new inclusive dates --- CumulusMX/QueryDayFile.cs | 13 ++++++++----- Updates.txt | 3 ++- 2 files changed, 10 insertions(+), 6 deletions(-) diff --git a/CumulusMX/QueryDayFile.cs b/CumulusMX/QueryDayFile.cs index f53f42b5..82c45ea6 100644 --- a/CumulusMX/QueryDayFile.cs +++ b/CumulusMX/QueryDayFile.cs @@ -11,6 +11,8 @@ using SQLite; +using static System.Runtime.InteropServices.JavaScript.JSType; + namespace CumulusMX { @@ -34,8 +36,6 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) throw new ArgumentException($"Invalid function name - '{function}'"); } - Program.cumulus.LogDebugMessage($"QueryDayFile: prop={propertyName}, func={function}, where={where}, from={from}, to={to}, resFunc={resfunc}"); - try { switch (from) @@ -47,19 +47,19 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) case "ThisMonth": fromDate = new DateTime(DateTime.Now.Year, DateTime.Now.Month, 1, 0, 0, 0, DateTimeKind.Local); - toDate = fromDate.AddMonths(1); + toDate = new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.DaysInMonth(DateTime.Now.Year, DateTime.Now.Month), 0, 0, 0, DateTimeKind.Local); break; case string s when s.StartsWith("Month-"): var rel = int.Parse(s.Split('-')[1]); fromDate = new DateTime(DateTime.Now.Year, DateTime.Now.Month, 1, 0, 0, 0, DateTimeKind.Local).AddMonths(-rel); - toDate = fromDate.AddMonths(1); + toDate = fromDate.AddMonths(1).AddDays(-1); break; case string s when s.StartsWith("Year-"): rel = int.Parse(s.Split('-')[1]); fromDate = new DateTime(DateTime.Now.Year, 1, 1, 0, 0, 0, DateTimeKind.Local).AddYears(-rel); - toDate = fromDate.AddYears(1); + toDate = fromDate.AddYears(1).AddDays(-1); break; case string s when s.Length > 5 && s.StartsWith("Month"): @@ -97,6 +97,9 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) throw new ArgumentException("Error parsing from/to dates: " + ex.Message); } + Program.cumulus.LogDebugMessage($"QueryDayFile: resFunc={resfunc}, prop={propertyName}, func={function}, where={where}, from={from}, to={to}, start={fromDate:yyyy-MM-dd}, end={toDate:yyyy-MM-dd}"); + + DateTime logTime = DateTime.MinValue; double value = -9999; diff --git a/Updates.txt b/Updates.txt index 18c8ed31..d9db59a3 100644 --- a/Updates.txt +++ b/Updates.txt @@ -74,7 +74,8 @@ New - NewRecordAlarm somewhat replicates the existing #newrecord web tag, but is also controlled by the alarm being enable/disabled - NewRecordAlarmMessage displays the last new record alarm text message - Old MD5 hash files are now deleted on startup -- New web tag <#DayFileQuery> which allows flexible querying of the day file. Please read the separate documention (QueryDayFile.md) for more details +- New web tag <#DayFileQuery> which allows flexible querying of the day file. + - Please read the separate documention (/MUtils/QueryDayFile.md) for more details - Added a script to /MXutils/linux/Fix_FineOffset_USB.sh to fix Fine Offset USB stations Changed From 8e64ff0722dbfedf16e82c63ded07a2933fc9e52 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sat, 21 Sep 2024 14:07:19 +0100 Subject: [PATCH 52/63] Suspend data during rollover processing --- CumulusMX/DavisCloudStation.cs | 5 + CumulusMX/DavisWllStation.cs | 17 +- CumulusMX/EcowittCloudStation.cs | 2 +- CumulusMX/EcowittHttpApiStation.cs | 277 +++++++++++++++-------------- CumulusMX/GW1000Station.cs | 115 ++++++------ CumulusMX/HttpStationEcowitt.cs | 9 +- CumulusMX/JsonStation.cs | 11 +- CumulusMX/QueryDayFile.cs | 10 +- CumulusMX/WeatherStation.cs | 7 + Updates.txt | 2 +- 10 files changed, 251 insertions(+), 204 deletions(-) diff --git a/CumulusMX/DavisCloudStation.cs b/CumulusMX/DavisCloudStation.cs index 8b791353..f89728d9 100644 --- a/CumulusMX/DavisCloudStation.cs +++ b/CumulusMX/DavisCloudStation.cs @@ -218,6 +218,11 @@ public override void Stop() private async void GetCurrent(object source, ElapsedEventArgs e) { + if (DayResetInProgress) + { + return; + } + cumulus.LogMessage("GetCurrent: Get WL.com Current Data"); if (cumulus.WllApiKey == string.Empty || cumulus.WllApiSecret == string.Empty) diff --git a/CumulusMX/DavisWllStation.cs b/CumulusMX/DavisWllStation.cs index 7069e1d2..8f93f0c7 100644 --- a/CumulusMX/DavisWllStation.cs +++ b/CumulusMX/DavisWllStation.cs @@ -285,8 +285,16 @@ public override void Start() // we get duplicate packets over IPv4 and IPv6, plus if the host has multiple interfaces to the local LAN if (!Utils.ByteArraysEqual(lastMessage, bcastTask.Result.Buffer)) { - var jsonStr = Encoding.UTF8.GetString(bcastTask.Result.Buffer); - DecodeBroadcast(jsonStr, bcastTask.Result.RemoteEndPoint); + if (!DayResetInProgress) + { + var jsonStr = Encoding.UTF8.GetString(bcastTask.Result.Buffer); + DecodeBroadcast(jsonStr, bcastTask.Result.RemoteEndPoint); + } + else + { + broadcastReceived = true; + cumulus.LogMessage("WLL: Rollover in progress, broadcast ignored"); + } lastMessage = bcastTask.Result.Buffer.ToArray(); } } @@ -465,6 +473,11 @@ private async void GetWllCurrent(object source, ElapsedEventArgs e) string ip; int retry = 1; + if (DayResetInProgress) + { + return; + } + lock (threadSafer) { ip = cumulus.DavisOptions.IPAddr; diff --git a/CumulusMX/EcowittCloudStation.cs b/CumulusMX/EcowittCloudStation.cs index ce7a1915..b6c93cf7 100644 --- a/CumulusMX/EcowittCloudStation.cs +++ b/CumulusMX/EcowittCloudStation.cs @@ -153,7 +153,7 @@ public override void Start() while (!cumulus.cancellationToken.IsCancellationRequested) { - if (DateTime.Now >= nextFetch) + if (DateTime.Now >= nextFetch && !DayResetInProgress) { try { diff --git a/CumulusMX/EcowittHttpApiStation.cs b/CumulusMX/EcowittHttpApiStation.cs index 5f32e083..a4a54310 100644 --- a/CumulusMX/EcowittHttpApiStation.cs +++ b/CumulusMX/EcowittHttpApiStation.cs @@ -153,182 +153,185 @@ public override void Start() while (!cumulus.cancellationToken.IsCancellationRequested) { - var rawData = localApi.GetLiveData(cumulus.cancellationToken); - if (rawData is not null) + if (!DayResetInProgress) { - dataLastRead = DateTime.Now; - - // process the common_list sensors - if (rawData.common_list != null) + var rawData = localApi.GetLiveData(cumulus.cancellationToken); + if (rawData is not null) { - ProcessCommonList(rawData.common_list, dataLastRead); - } + dataLastRead = DateTime.Now; - // process base station values - if (rawData.wh25 != null) - { - ProcessWh25(rawData.wh25, dataLastRead); - } + // process the common_list sensors + if (rawData.common_list != null) + { + ProcessCommonList(rawData.common_list, dataLastRead); + } - // process rain values - if (cumulus.Gw1000PrimaryRainSensor == 0 && rawData.rain != null) - { - ProcessRain(rawData.rain, false); - } - - if ((cumulus.Gw1000PrimaryRainSensor == 1 || (cumulus.Gw1000PrimaryRainSensor == 0 && cumulus.EcowittIsRainingUsePiezo)) && rawData.piezoRain != null) - { - // if we are using piezo as the primary rain sensor - // or using the tipper at the primary, but want to use the piezo srain value for IsRaining - ProcessRain(rawData.piezoRain, cumulus.Gw1000PrimaryRainSensor == 0 && cumulus.EcowittIsRainingUsePiezo); - } + // process base station values + if (rawData.wh25 != null) + { + ProcessWh25(rawData.wh25, dataLastRead); + } - if (rawData.lightning != null) - { - ProcessLightning(rawData.lightning, dataLastRead); - } + // process rain values + if (cumulus.Gw1000PrimaryRainSensor == 0 && rawData.rain != null) + { + ProcessRain(rawData.rain, false); + } - if (rawData.co2 != null) - { - ProcessCo2(rawData.co2); - } + if ((cumulus.Gw1000PrimaryRainSensor == 1 || (cumulus.Gw1000PrimaryRainSensor == 0 && cumulus.EcowittIsRainingUsePiezo)) && rawData.piezoRain != null) + { + // if we are using piezo as the primary rain sensor + // or using the tipper at the primary, but want to use the piezo srain value for IsRaining + ProcessRain(rawData.piezoRain, cumulus.Gw1000PrimaryRainSensor == 0 && cumulus.EcowittIsRainingUsePiezo); + } - if (rawData.ch_pm25 != null) - { - ProcessChPm25(rawData.ch_pm25); - } + if (rawData.lightning != null) + { + ProcessLightning(rawData.lightning, dataLastRead); + } - if (rawData.ch_leak != null) - { - ProcessLeak(rawData.ch_leak); - } + if (rawData.co2 != null) + { + ProcessCo2(rawData.co2); + } - if (rawData.ch_aisle != null) - { - ProcessExtraTempHum(rawData.ch_aisle, dataLastRead); - } + if (rawData.ch_pm25 != null) + { + ProcessChPm25(rawData.ch_pm25); + } - if (rawData.ch_temp != null) - { - ProcessUserTemp(rawData.ch_temp); - } + if (rawData.ch_leak != null) + { + ProcessLeak(rawData.ch_leak); + } - if (rawData.ch_soil != null) - { - ProcessSoilMoisture(rawData.ch_soil); - } + if (rawData.ch_aisle != null) + { + ProcessExtraTempHum(rawData.ch_aisle, dataLastRead); + } - if (rawData.ch_leaf != null) - { - ProcessLeafWet(rawData.ch_leaf); - } + if (rawData.ch_temp != null) + { + ProcessUserTemp(rawData.ch_temp); + } - // Now do the stuff that requires more than one input parameter + if (rawData.ch_soil != null) + { + ProcessSoilMoisture(rawData.ch_soil); + } - // Only set the lightning time/distance if it is newer than what we already have - the GW1000 seems to reset this value - if (newLightningTime > LightningTime) - { - LightningTime = newLightningTime; - if (newLightningDistance < 999) - LightningDistance = newLightningDistance; - } + if (rawData.ch_leaf != null) + { + ProcessLeafWet(rawData.ch_leaf); + } - // Process outdoor temperature here, as GW1000 currently does not supply Dew Point so we have to calculate it in DoOutdoorTemp() - if (outdoortemp > -999) - DoOutdoorTemp(ConvertUnits.TempCToUser(outdoortemp), dataLastRead); + // Now do the stuff that requires more than one input parameter - // Same for extra T/H sensors - for (var i = 1; i <= 8; i++) - { - if (ExtraHum[i] > 0) + // Only set the lightning time/distance if it is newer than what we already have - the GW1000 seems to reset this value + if (newLightningTime > LightningTime) { - var dp = MeteoLib.DewPoint(ConvertUnits.UserTempToC(ExtraTemp[i]), ExtraHum[i]); - ExtraDewPoint[i] = ConvertUnits.TempCToUser(dp); + LightningTime = newLightningTime; + if (newLightningDistance < 999) + LightningDistance = newLightningDistance; } - } - if (gustLast > -999 && windSpeedLast > -999 && windDirLast > -999) - { - DoWind(gustLast, windDirLast, windSpeedLast, dataLastRead); - } + // Process outdoor temperature here, as GW1000 currently does not supply Dew Point so we have to calculate it in DoOutdoorTemp() + if (outdoortemp > -999) + DoOutdoorTemp(ConvertUnits.TempCToUser(outdoortemp), dataLastRead); - if (rainLast > -999 && rainRateLast > -999) - { - DoRain(rainLast, rainRateLast, dataLastRead); - } + // Same for extra T/H sensors + for (var i = 1; i <= 8; i++) + { + if (ExtraHum[i] > 0) + { + var dp = MeteoLib.DewPoint(ConvertUnits.UserTempToC(ExtraTemp[i]), ExtraHum[i]); + ExtraDewPoint[i] = ConvertUnits.TempCToUser(dp); + } + } - if (outdoortemp > -999) - { - DoWindChill(windchill, dataLastRead); - DoApparentTemp(dataLastRead); - DoFeelsLike(dataLastRead); - DoHumidex(dataLastRead); - DoCloudBaseHeatIndex(dataLastRead); + if (gustLast > -999 && windSpeedLast > -999 && windDirLast > -999) + { + DoWind(gustLast, windDirLast, windSpeedLast, dataLastRead); + } - if (cumulus.StationOptions.CalculateSLP) + if (rainLast > -999 && rainRateLast > -999) { - var abs = cumulus.Calib.Press.Calibrate(StationPressure); - var slp = MeteoLib.GetSeaLevelPressure(AltitudeM(cumulus.Altitude), ConvertUnits.UserPressToMB(abs), ConvertUnits.UserTempToC(OutdoorTemperature), cumulus.Latitude); - DoPressure(ConvertUnits.PressMBToUser(slp), dataLastRead); + DoRain(rainLast, rainRateLast, dataLastRead); } - } - DoForecast("", false); + if (outdoortemp > -999) + { + DoWindChill(windchill, dataLastRead); + DoApparentTemp(dataLastRead); + DoFeelsLike(dataLastRead); + DoHumidex(dataLastRead); + DoCloudBaseHeatIndex(dataLastRead); - cumulus.BatteryLowAlarm.Triggered = batteryLow; + if (cumulus.StationOptions.CalculateSLP) + { + var abs = cumulus.Calib.Press.Calibrate(StationPressure); + var slp = MeteoLib.GetSeaLevelPressure(AltitudeM(cumulus.Altitude), ConvertUnits.UserPressToMB(abs), ConvertUnits.UserTempToC(OutdoorTemperature), cumulus.Latitude); + DoPressure(ConvertUnits.PressMBToUser(slp), dataLastRead); + } + } - UpdateStatusPanel(dataLastRead); - UpdateMQTT(); + DoForecast("", false); - dataReceived = true; - DataStopped = false; - cumulus.DataStoppedAlarm.Triggered = false; - LastDataReadTime = dataLastRead; + cumulus.BatteryLowAlarm.Triggered = batteryLow; - var minute = DateTime.Now.Minute; - if (minute != lastMinute) - { - lastMinute = minute; + UpdateStatusPanel(dataLastRead); + UpdateMQTT(); - // at the start of every 20 minutes to trigger battery status check - if ((minute % 20) == 0 && !cumulus.cancellationToken.IsCancellationRequested) - { - _ = GetSensorIds(); - } + dataReceived = true; + DataStopped = false; + cumulus.DataStoppedAlarm.Triggered = false; + LastDataReadTime = dataLastRead; - // every day dump the clock drift at midday each day - if (minute == 0 && DateTime.Now.Hour == 12) + var minute = DateTime.Now.Minute; + if (minute != lastMinute) { - GetSystemInfo(true); - } + lastMinute = minute; - var hour = DateTime.Now.Hour; - if (lastHour != hour) - { - lastHour = hour; + // at the start of every 20 minutes to trigger battery status check + if ((minute % 20) == 0 && !cumulus.cancellationToken.IsCancellationRequested) + { + _ = GetSensorIds(); + } - if (hour == 13) + // every day dump the clock drift at midday each day + if (minute == 0 && DateTime.Now.Hour == 12) { - var fw = GetFirmwareVersion(); - if (fw != "???") - { - GW1000FirmwareVersion = fw; - deviceModel = GW1000FirmwareVersion.Split('_')[0]; - deviceFirmware = GW1000FirmwareVersion.Split('_')[1]; + GetSystemInfo(true); + } - var fwString = GW1000FirmwareVersion.Split(underscoreV, StringSplitOptions.None); - if (fwString.Length > 1) - { - fwVersion = new Version(fwString[1]); - } - else + var hour = DateTime.Now.Hour; + if (lastHour != hour) + { + lastHour = hour; + + if (hour == 13) + { + var fw = GetFirmwareVersion(); + if (fw != "???") { - // failed to get the version, lets assume it's fairly new - fwVersion = new Version("1.6.5"); + GW1000FirmwareVersion = fw; + deviceModel = GW1000FirmwareVersion.Split('_')[0]; + deviceFirmware = GW1000FirmwareVersion.Split('_')[1]; + + var fwString = GW1000FirmwareVersion.Split(underscoreV, StringSplitOptions.None); + if (fwString.Length > 1) + { + fwVersion = new Version(fwString[1]); + } + else + { + // failed to get the version, lets assume it's fairly new + fwVersion = new Version("1.6.5"); + } } - } - _ = CheckAvailableFirmware(); + _ = CheckAvailableFirmware(); + } } } } diff --git a/CumulusMX/GW1000Station.cs b/CumulusMX/GW1000Station.cs index 67b76e2e..a24623e6 100644 --- a/CumulusMX/GW1000Station.cs +++ b/CumulusMX/GW1000Station.cs @@ -147,82 +147,85 @@ public override void Start() while (!cumulus.cancellationToken.IsCancellationRequested) { - if (Api.Connected) + if (!DayResetInProgress) { - GetLiveData(); - dataLastRead = DateTime.Now; - - // every 30 seconds read the rain rate - if ((cumulus.Gw1000PrimaryRainSensor == 1 || cumulus.StationOptions.UseRainForIsRaining == 2) && (DateTime.UtcNow - piezoLastRead).TotalSeconds >= 30 && !cumulus.cancellationToken.IsCancellationRequested) - { - GetPiezoRainData(); - piezoLastRead = DateTime.UtcNow; - } - - var minute = DateTime.Now.Minute; - if (minute != lastMinute) + if (Api.Connected) { - lastMinute = minute; + GetLiveData(); + dataLastRead = DateTime.Now; - // at the start of every 20 minutes to trigger battery status check - if ((minute % 20) == 0 && !cumulus.cancellationToken.IsCancellationRequested) + // every 30 seconds read the rain rate + if ((cumulus.Gw1000PrimaryRainSensor == 1 || cumulus.StationOptions.UseRainForIsRaining == 2) && (DateTime.UtcNow - piezoLastRead).TotalSeconds >= 30 && !cumulus.cancellationToken.IsCancellationRequested) { - GetSensorIdsNew(); + GetPiezoRainData(); + piezoLastRead = DateTime.UtcNow; } - // every day dump the clock drift at midday each day - if (minute == 0 && DateTime.Now.Hour == 12) + var minute = DateTime.Now.Minute; + if (minute != lastMinute) { - GetSystemInfo(true); - } + lastMinute = minute; - var hour = DateTime.Now.Hour; - if (lastHour != hour) - { - lastHour = hour; + // at the start of every 20 minutes to trigger battery status check + if ((minute % 20) == 0 && !cumulus.cancellationToken.IsCancellationRequested) + { + GetSensorIdsNew(); + } - if (hour == 13) + // every day dump the clock drift at midday each day + if (minute == 0 && DateTime.Now.Hour == 12) { - var fw = GetFirmwareVersion(); - if (fw != "???") - { - GW1000FirmwareVersion = fw; - deviceModel = GW1000FirmwareVersion.Split('_')[0]; - deviceFirmware = GW1000FirmwareVersion.Split('_')[1]; + GetSystemInfo(true); + } - var fwString = GW1000FirmwareVersion.Split(underscoreV, StringSplitOptions.None); - if (fwString.Length > 1) - { - fwVersion = new Version(fwString[1]); - } - else + var hour = DateTime.Now.Hour; + if (lastHour != hour) + { + lastHour = hour; + + if (hour == 13) + { + var fw = GetFirmwareVersion(); + if (fw != "???") { - // failed to get the version, lets assume it's fairly new - fwVersion = new Version("1.6.5"); + GW1000FirmwareVersion = fw; + deviceModel = GW1000FirmwareVersion.Split('_')[0]; + deviceFirmware = GW1000FirmwareVersion.Split('_')[1]; + + var fwString = GW1000FirmwareVersion.Split(underscoreV, StringSplitOptions.None); + if (fwString.Length > 1) + { + fwVersion = new Version(fwString[1]); + } + else + { + // failed to get the version, lets assume it's fairly new + fwVersion = new Version("1.6.5"); + } } - } - _ = CheckAvailableFirmware(); + _ = CheckAvailableFirmware(); + } } } } - } - else - { - cumulus.LogMessage("Attempting to reconnect to Ecowitt device..."); - Api.OpenTcpPort(cumulus.Gw1000IpAddress, AtPort); - if (Api.Connected) - { - cumulus.LogMessage("Reconnected to Ecowitt device"); - GetLiveData(); - } else { - // add a small extra delay before trying again - cumulus.LogMessage("Delaying before attempting reconnect"); - if (cumulus.cancellationToken.WaitHandle.WaitOne(TimeSpan.FromMilliseconds(20000))) + cumulus.LogMessage("Attempting to reconnect to Ecowitt device..."); + Api.OpenTcpPort(cumulus.Gw1000IpAddress, AtPort); + if (Api.Connected) { - break; + cumulus.LogMessage("Reconnected to Ecowitt device"); + GetLiveData(); + } + else + { + // add a small extra delay before trying again + cumulus.LogMessage("Delaying before attempting reconnect"); + if (cumulus.cancellationToken.WaitHandle.WaitOne(TimeSpan.FromMilliseconds(20000))) + { + break; + } } } } diff --git a/CumulusMX/HttpStationEcowitt.cs b/CumulusMX/HttpStationEcowitt.cs index 567f38c1..a177b57c 100644 --- a/CumulusMX/HttpStationEcowitt.cs +++ b/CumulusMX/HttpStationEcowitt.cs @@ -307,9 +307,16 @@ public string ProcessData(IHttpContext context, bool main, DateTime? ts = null) var procName = main ? "ProcessData" : "ProcessExtraData"; + if (DayResetInProgress) + { + cumulus.LogMessage("ProcessData: Rollover in progress, incoming data ignored"); + context.Response.StatusCode = 200; + return "success"; + } + if (starting || stopping) { - cumulus.LogMessage($"Station {(starting ? "starting" : "stopping")}, incoming data ignored"); + cumulus.LogMessage($"ProcessData: Station {(starting ? "starting" : "stopping")}, incoming data ignored"); context.Response.StatusCode = 200; return "success"; } diff --git a/CumulusMX/JsonStation.cs b/CumulusMX/JsonStation.cs index 0c79668a..c2c93557 100644 --- a/CumulusMX/JsonStation.cs +++ b/CumulusMX/JsonStation.cs @@ -113,11 +113,10 @@ private void GetDataFromFile() watcher = new FileSystemWatcher(fileInfo.DirectoryName) { - NotifyFilter = NotifyFilters.LastWrite + NotifyFilter = NotifyFilters.LastWrite, + Filter = fileInfo.Name }; - watcher.Filter = fileInfo.Name; - watcher.Changed += OnFileChanged; watcher.Created += OnFileChanged; watcher.EnableRaisingEvents = true; @@ -205,6 +204,12 @@ public void ReceiveDataFromMqtt(MQTTnet.MqttApplicationMessage appMessage) private string ApplyData(string dataString) { + if (DayResetInProgress) + { + cumulus.LogMessage("ApplyData: Day reset in progress, ignoring incoming data"); + return string.Empty; + } + var retStr = new StringBuilder(); var data = dataString.FromJson(); diff --git a/CumulusMX/QueryDayFile.cs b/CumulusMX/QueryDayFile.cs index 82c45ea6..3a3d6818 100644 --- a/CumulusMX/QueryDayFile.cs +++ b/CumulusMX/QueryDayFile.cs @@ -206,7 +206,9 @@ internal class QueryDayFile(SQLiteConnection databaseConnection) } } } - Program.cumulus.LogDebugMessage($"QueryDayFile: Result = [{value:0.00}, {logTime:g}]"); + + var valStr = value.ToString("0.000", inv); + Program.cumulus.LogDebugMessage($"QueryDayFile: Result = [{valStr}, {logTime:g}]"); return (value, logTime); } @@ -244,12 +246,14 @@ public string WebQuery(IHttpContext context) return "{\"value\": \"n/a\", \"time\": \"n/a\"}"; } + var valStr = value.ToString("0.000", inv); + if (time == DateTime.MinValue) { - return $"{{\"value\": {value:0.000}, \"time\":\"n/a\"}}"; + return $"{{\"value\": {valStr}, \"time\":\"n/a\"}}"; } - return $"{{\"value\": {value:0.000}, \"time\":\"{time.ToString(format)}\"}}"; + return $"{{\"value\": {valStr}, \"time\":\"{time.ToString(format)}\"}}"; } catch (Exception ex) { diff --git a/CumulusMX/WeatherStation.cs b/CumulusMX/WeatherStation.cs index bf679dd4..f99ba382 100644 --- a/CumulusMX/WeatherStation.cs +++ b/CumulusMX/WeatherStation.cs @@ -210,6 +210,8 @@ public class DayFileRec public bool gotraindaystart = false; protected double prevraincounter = 0.0; + protected bool DayResetInProgress = false; + public struct DailyHighLow { public double HighGust; @@ -6890,6 +6892,9 @@ public void DayReset(DateTime timestamp) int drday = timestamp.Day; DateTime yesterday = timestamp.AddDays(-1); cumulus.LogMessage("=== Day reset, today = " + drday); + + DayResetInProgress = true; + if (drday != DayResetDay) { cumulus.LogMessage("=== Day reset for " + yesterday.Date); @@ -7613,6 +7618,8 @@ public void DayReset(DateTime timestamp) CurrentDay = timestamp.Day; CurrentMonth = timestamp.Month; CurrentYear = timestamp.Year; + DayResetInProgress = false; + cumulus.LogMessage("=== Day reset complete"); cumulus.LogMessage("Now recording data for day=" + CurrentDay + " month=" + CurrentMonth + " year=" + CurrentYear); } diff --git a/Updates.txt b/Updates.txt index d9db59a3..b67fb299 100644 --- a/Updates.txt +++ b/Updates.txt @@ -75,7 +75,7 @@ New - NewRecordAlarmMessage displays the last new record alarm text message - Old MD5 hash files are now deleted on startup - New web tag <#DayFileQuery> which allows flexible querying of the day file. - - Please read the separate documention (/MUtils/QueryDayFile.md) for more details + - Please read the separate documention (/MXutils/QueryDayFile.md) for more details - Added a script to /MXutils/linux/Fix_FineOffset_USB.sh to fix Fine Offset USB stations Changed From 74e70d2d0f635ee714d5081dcfb12f899ade7d37 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sat, 21 Sep 2024 14:29:37 +0100 Subject: [PATCH 53/63] Fix Davis cloud station continually attempting to download historic data on error. --- CumulusMX/DavisCloudStation.cs | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/CumulusMX/DavisCloudStation.cs b/CumulusMX/DavisCloudStation.cs index f89728d9..15a2b79c 100644 --- a/CumulusMX/DavisCloudStation.cs +++ b/CumulusMX/DavisCloudStation.cs @@ -500,7 +500,7 @@ private void GetHistoricData(BackgroundWorker worker) cumulus.LogErrorMessage($"GetHistoricData: WeatherLink API Historic Error: {historyError.code}, {historyError.message}"); Cumulus.LogConsoleMessage($" - Error {historyError.code}: {historyError.message}", ConsoleColor.Red); //cumulus.LastUpdateTime = Utils.FromUnixTime(endTime) - maxArchiveRuns = 0; + maxArchiveRuns = -1; return; } @@ -509,7 +509,7 @@ private void GetHistoricData(BackgroundWorker worker) cumulus.LogWarningMessage("GetHistoricData: WeatherLink API Historic: No data was returned. Check your Device Id."); Cumulus.LogConsoleMessage(" - No historic data available"); lastHistoricData = Utils.FromUnixTime(endTime); - maxArchiveRuns = 0; + maxArchiveRuns = -1; return; } else if (responseBody.StartsWith("{\"")) // basic sanity check @@ -550,7 +550,7 @@ private void GetHistoricData(BackgroundWorker worker) cumulus.LogErrorMessage("GetHistoricData: Invalid historic message received"); cumulus.LogMessage("GetHistoricData: Received: " + responseBody); lastHistoricData = Utils.FromUnixTime(endTime); - maxArchiveRuns = 0; + maxArchiveRuns = -1; return; } } @@ -564,7 +564,7 @@ private void GetHistoricData(BackgroundWorker worker) } lastHistoricData = Utils.FromUnixTime(endTime); - maxArchiveRuns = 0; + maxArchiveRuns = -1; return; } From 445618a713c77e8d2f82b06a52d948c2ba66557b Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sat, 21 Sep 2024 14:33:35 +0100 Subject: [PATCH 54/63] Updates.txt --- CumulusMX/DavisCloudStation.cs | 2 +- Updates.txt | 2 ++ 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/CumulusMX/DavisCloudStation.cs b/CumulusMX/DavisCloudStation.cs index 15a2b79c..ea2429c0 100644 --- a/CumulusMX/DavisCloudStation.cs +++ b/CumulusMX/DavisCloudStation.cs @@ -3780,7 +3780,7 @@ private void SetDataTimeout(string subscription) _ => 15 + 3, }; - cumulus.LogMessage($"GetStations: Subscription type = {subscription}, data timeout = {DataTimeoutMins} minutes"); + cumulus.LogMessage($"GetStations: Subscription type = {subscriptionLevel}, data timeout = {DataTimeoutMins} minutes"); } private void GetAvailableSensors() diff --git a/Updates.txt b/Updates.txt index b67fb299..b8c5c7c0 100644 --- a/Updates.txt +++ b/Updates.txt @@ -6,6 +6,7 @@ BETA 4037 Changes: - Station Settings for HTTP API station now drops MAC to match Wizard settings - Query Daily Data to date now inclusive - Daily data for query now added to at rollover, and amended at day file edits +- Fix Davis Cloud Station continually attempting to download history data on error BETA 4036 Changes: @@ -93,6 +94,7 @@ Fixed - Soil moisture units now follow the source - Example if you have a main station Davis with sensors 1 & 2, their units will be cb, and you have Ecowitt extra sensors 3 & 4, their units will be % - This does mean a change to the Units JSON and the graph scripts for the default web site. A re-upload of /js/cumuluscharts.js and /js/selectachart.js will be required +- Fix Davis Cloud Station continually attempting to download history data on error Package Updates From a45b62e99017305724b17962d3d9a67efe967f0d Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sat, 21 Sep 2024 20:54:30 +0100 Subject: [PATCH 55/63] Fix a lurking problem with the today.ini and yesterday.ini files that has been there from day 0 --- CumulusMX/WeatherStation.cs | 94 ++++++++++++++++++------------------- Updates.txt | 3 +- 2 files changed, 49 insertions(+), 48 deletions(-) diff --git a/CumulusMX/WeatherStation.cs b/CumulusMX/WeatherStation.cs index f99ba382..a82e107c 100644 --- a/CumulusMX/WeatherStation.cs +++ b/CumulusMX/WeatherStation.cs @@ -1023,15 +1023,15 @@ public void WriteTodayFile(DateTime timestamp, bool Log) // Date ini.SetValue("General", "Date", timestamp.AddHours(hourInc).ToShortDateString()); // Timestamp - ini.SetValue("General", "Timestamp", cumulus.LastUpdateTime.ToString("s")); + ini.SetValue("General", "Timestamp", cumulus.LastUpdateTime); ini.SetValue("General", "CurrentYear", CurrentYear); ini.SetValue("General", "CurrentMonth", CurrentMonth); ini.SetValue("General", "CurrentDay", CurrentDay); // Wind ini.SetValue("Wind", "Speed", HiLoToday.HighWind); - ini.SetValue("Wind", "SpTime", HiLoToday.HighWindTime.ToString("HH:mm")); + ini.SetValue("Wind", "SpTime", HiLoToday.HighWindTime); ini.SetValue("Wind", "Gust", HiLoToday.HighGust); - ini.SetValue("Wind", "Time", HiLoToday.HighGustTime.ToString("HH:mm")); + ini.SetValue("Wind", "Time", HiLoToday.HighGustTime); ini.SetValue("Wind", "Bearing", HiLoToday.HighGustBearing); ini.SetValue("Wind", "Direction", CompassPoint(HiLoToday.HighGustBearing)); ini.SetValue("Wind", "Windrun", WindRunToday); @@ -1041,9 +1041,9 @@ public void WriteTodayFile(DateTime timestamp, bool Log) ini.SetValue("Wind", "DominantWindBearingY", DominantWindBearingY); // Temperature ini.SetValue("Temp", "Low", HiLoToday.LowTemp); - ini.SetValue("Temp", "LTime", HiLoToday.LowTempTime.ToString("HH:mm")); + ini.SetValue("Temp", "LTime", HiLoToday.LowTempTime); ini.SetValue("Temp", "High", HiLoToday.HighTemp); - ini.SetValue("Temp", "HTime", HiLoToday.HighTempTime.ToString("HH:mm")); + ini.SetValue("Temp", "HTime", HiLoToday.HighTempTime); ini.SetValue("Temp", "Total", TempTotalToday); ini.SetValue("Temp", "Samples", tempsamplestoday); ini.SetValue("Temp", "ChillHours", ChillHours); @@ -1058,16 +1058,16 @@ public void WriteTodayFile(DateTime timestamp, bool Log) ini.SetValue("TempMidnight", "HTime", HiLoTodayMidnight.HighTempTime); // Pressure ini.SetValue("Pressure", "Low", HiLoToday.LowPress); - ini.SetValue("Pressure", "LTime", HiLoToday.LowPressTime.ToString("HH:mm")); + ini.SetValue("Pressure", "LTime", HiLoToday.LowPressTime); ini.SetValue("Pressure", "High", HiLoToday.HighPress); - ini.SetValue("Pressure", "HTime", HiLoToday.HighPressTime.ToString("HH:mm")); + ini.SetValue("Pressure", "HTime", HiLoToday.HighPressTime); // rain ini.SetValue("Rain", "High", HiLoToday.HighRainRate); - ini.SetValue("Rain", "HTime", HiLoToday.HighRainRateTime.ToString("HH:mm")); + ini.SetValue("Rain", "HTime", HiLoToday.HighRainRateTime); ini.SetValue("Rain", "HourlyHigh", HiLoToday.HighHourlyRain); - ini.SetValue("Rain", "HHourlyTime", HiLoToday.HighHourlyRainTime.ToString("HH:mm")); + ini.SetValue("Rain", "HHourlyTime", HiLoToday.HighHourlyRainTime); ini.SetValue("Rain", "High24h", HiLoToday.HighRain24h); - ini.SetValue("Rain", "High24hTime", HiLoToday.HighRain24hTime.ToString("HH:mm")); + ini.SetValue("Rain", "High24hTime", HiLoToday.HighRain24hTime); ini.SetValue("Rain", "Yesterday", RainYesterday); ini.SetValue("Rain", "Start", RainCounterDayStart); ini.SetValue("Rain", "Midnight", MidnightRainCount); @@ -1082,35 +1082,35 @@ public void WriteTodayFile(DateTime timestamp, bool Log) // humidity ini.SetValue("Humidity", "Low", HiLoToday.LowHumidity); ini.SetValue("Humidity", "High", HiLoToday.HighHumidity); - ini.SetValue("Humidity", "LTime", HiLoToday.LowHumidityTime.ToString("HH:mm")); - ini.SetValue("Humidity", "HTime", HiLoToday.HighHumidityTime.ToString("HH:mm")); + ini.SetValue("Humidity", "LTime", HiLoToday.LowHumidityTime); + ini.SetValue("Humidity", "HTime", HiLoToday.HighHumidityTime); // Solar ini.SetValue("Solar", "SunshineHours", SunshineHours); ini.SetValue("Solar", "SunshineHoursToMidnight", SunshineToMidnight); // heat index ini.SetValue("HeatIndex", "High", HiLoToday.HighHeatIndex); - ini.SetValue("HeatIndex", "HTime", HiLoToday.HighHeatIndexTime.ToString("HH:mm")); + ini.SetValue("HeatIndex", "HTime", HiLoToday.HighHeatIndexTime); // App temp ini.SetValue("AppTemp", "Low", HiLoToday.LowAppTemp); - ini.SetValue("AppTemp", "LTime", HiLoToday.LowAppTempTime.ToString("HH:mm")); + ini.SetValue("AppTemp", "LTime", HiLoToday.LowAppTempTime); ini.SetValue("AppTemp", "High", HiLoToday.HighAppTemp); - ini.SetValue("AppTemp", "HTime", HiLoToday.HighAppTempTime.ToString("HH:mm")); + ini.SetValue("AppTemp", "HTime", HiLoToday.HighAppTempTime); // Feels like ini.SetValue("FeelsLike", "Low", HiLoToday.LowFeelsLike); - ini.SetValue("FeelsLike", "LTime", HiLoToday.LowFeelsLikeTime.ToString("HH:mm")); + ini.SetValue("FeelsLike", "LTime", HiLoToday.LowFeelsLikeTime); ini.SetValue("FeelsLike", "High", HiLoToday.HighFeelsLike); - ini.SetValue("FeelsLike", "HTime", HiLoToday.HighFeelsLikeTime.ToString("HH:mm")); + ini.SetValue("FeelsLike", "HTime", HiLoToday.HighFeelsLikeTime); // Humidex ini.SetValue("Humidex", "High", HiLoToday.HighHumidex); - ini.SetValue("Humidex", "HTime", HiLoToday.HighHumidexTime.ToString("HH:mm")); + ini.SetValue("Humidex", "HTime", HiLoToday.HighHumidexTime); // wind chill ini.SetValue("WindChill", "Low", HiLoToday.LowWindChill); - ini.SetValue("WindChill", "LTime", HiLoToday.LowWindChillTime.ToString("HH:mm")); + ini.SetValue("WindChill", "LTime", HiLoToday.LowWindChillTime); // Dewpoint ini.SetValue("Dewpoint", "Low", HiLoToday.LowDewPoint); - ini.SetValue("Dewpoint", "LTime", HiLoToday.LowDewPointTime.ToString("HH:mm")); + ini.SetValue("Dewpoint", "LTime", HiLoToday.LowDewPointTime); ini.SetValue("Dewpoint", "High", HiLoToday.HighDewPoint); - ini.SetValue("Dewpoint", "HTime", HiLoToday.HighDewPointTime.ToString("HH:mm")); + ini.SetValue("Dewpoint", "HTime", HiLoToday.HighDewPointTime); // NOAA report names ini.SetValue("NOAA", "LatestMonthlyReport", cumulus.NOAAconf.LatestMonthReport); @@ -1118,9 +1118,9 @@ public void WriteTodayFile(DateTime timestamp, bool Log) // Solar ini.SetValue("Solar", "HighSolarRad", HiLoToday.HighSolar); - ini.SetValue("Solar", "HighSolarRadTime", HiLoToday.HighSolarTime.ToString("HH:mm")); + ini.SetValue("Solar", "HighSolarRadTime", HiLoToday.HighSolarTime); ini.SetValue("Solar", "HighUV", HiLoToday.HighUv); - ini.SetValue("Solar", "HighUVTime", HiLoToday.HighUvTime.ToString("HH:mm")); + ini.SetValue("Solar", "HighUVTime", HiLoToday.HighUvTime); ini.SetValue("Solar", "SunStart", StartOfDaySunHourCounter); // Special Fine Offset data @@ -6732,76 +6732,76 @@ public void WriteYesterdayFile(DateTime logdate) ini.SetValue("General", "Date", logdate.AddHours(hourInc)); // Wind ini.SetValue("Wind", "Speed", HiLoYest.HighWind); - ini.SetValue("Wind", "SpTime", HiLoYest.HighWindTime.ToString("HH:mm")); + ini.SetValue("Wind", "SpTime", HiLoYest.HighWindTime); ini.SetValue("Wind", "Gust", HiLoYest.HighGust); - ini.SetValue("Wind", "Time", HiLoYest.HighGustTime.ToString("HH:mm")); + ini.SetValue("Wind", "Time", HiLoYest.HighGustTime); ini.SetValue("Wind", "Bearing", HiLoYest.HighGustBearing); ini.SetValue("Wind", "Direction", CompassPoint(HiLoYest.HighGustBearing)); ini.SetValue("Wind", "Windrun", YesterdayWindRun); ini.SetValue("Wind", "DominantWindBearing", YestDominantWindBearing); // Temperature ini.SetValue("Temp", "Low", HiLoYest.LowTemp); - ini.SetValue("Temp", "LTime", HiLoYest.LowTempTime.ToString("HH:mm")); + ini.SetValue("Temp", "LTime", HiLoYest.LowTempTime); ini.SetValue("Temp", "High", HiLoYest.HighTemp); - ini.SetValue("Temp", "HTime", HiLoYest.HighTempTime.ToString("HH:mm")); + ini.SetValue("Temp", "HTime", HiLoYest.HighTempTime); ini.SetValue("Temp", "ChillHours", YestChillHours); ini.SetValue("Temp", "HeatingDegreeDays", YestHeatingDegreeDays); ini.SetValue("Temp", "CoolingDegreeDays", YestCoolingDegreeDays); ini.SetValue("Temp", "AvgTemp", YestAvgTemp); // Temperature midnight ini.SetValue("TempMidnight", "Low", HiLoYestMidnight.LowTemp); - ini.SetValue("TempMidnight", "LTime", HiLoYestMidnight.LowTempTime.ToString("HH:mm")); + ini.SetValue("TempMidnight", "LTime", HiLoYestMidnight.LowTempTime); ini.SetValue("TempMidnight", "High", HiLoYestMidnight.HighTemp); - ini.SetValue("TempMidnight", "HTime", HiLoYestMidnight.HighTempTime.ToString("HH:mm")); + ini.SetValue("TempMidnight", "HTime", HiLoYestMidnight.HighTempTime); // Pressure ini.SetValue("Pressure", "Low", HiLoYest.LowPress); - ini.SetValue("Pressure", "LTime", HiLoYest.LowPressTime.ToString("HH:mm")); + ini.SetValue("Pressure", "LTime", HiLoYest.LowPressTime); ini.SetValue("Pressure", "High", HiLoYest.HighPress); - ini.SetValue("Pressure", "HTime", HiLoYest.HighPressTime.ToString("HH:mm")); + ini.SetValue("Pressure", "HTime", HiLoYest.HighPressTime); // rain ini.SetValue("Rain", "High", HiLoYest.HighRainRate); - ini.SetValue("Rain", "HTime", HiLoYest.HighRainRateTime.ToString("HH:mm")); + ini.SetValue("Rain", "HTime", HiLoYest.HighRainRateTime); ini.SetValue("Rain", "HourlyHigh", HiLoYest.HighHourlyRain); - ini.SetValue("Rain", "HHourlyTime", HiLoYest.HighHourlyRainTime.ToString("HH:mm")); + ini.SetValue("Rain", "HHourlyTime", HiLoYest.HighHourlyRainTime); ini.SetValue("Rain", "High24h", HiLoYest.HighRain24h); - ini.SetValue("Rain", "High24hTime", HiLoYest.HighRain24hTime.ToString("HH:mm")); + ini.SetValue("Rain", "High24hTime", HiLoYest.HighRain24hTime); ini.SetValue("Rain", "RG11Yesterday", RG11RainYesterday); // humidity ini.SetValue("Humidity", "Low", HiLoYest.LowHumidity); ini.SetValue("Humidity", "High", HiLoYest.HighHumidity); - ini.SetValue("Humidity", "LTime", HiLoYest.LowHumidityTime.ToString("HH:mm")); - ini.SetValue("Humidity", "HTime", HiLoYest.HighHumidityTime.ToString("HH:mm")); + ini.SetValue("Humidity", "LTime", HiLoYest.LowHumidityTime); + ini.SetValue("Humidity", "HTime", HiLoYest.HighHumidityTime); // Solar ini.SetValue("Solar", "SunshineHours", YestSunshineHours); // heat index ini.SetValue("HeatIndex", "High", HiLoYest.HighHeatIndex); - ini.SetValue("HeatIndex", "HTime", HiLoYest.HighHeatIndexTime.ToString("HH:mm")); + ini.SetValue("HeatIndex", "HTime", HiLoYest.HighHeatIndexTime); // App temp ini.SetValue("AppTemp", "Low", HiLoYest.LowAppTemp); - ini.SetValue("AppTemp", "LTime", HiLoYest.LowAppTempTime.ToString("HH:mm")); + ini.SetValue("AppTemp", "LTime", HiLoYest.LowAppTempTime); ini.SetValue("AppTemp", "High", HiLoYest.HighAppTemp); - ini.SetValue("AppTemp", "HTime", HiLoYest.HighAppTempTime.ToString("HH:mm")); + ini.SetValue("AppTemp", "HTime", HiLoYest.HighAppTempTime); // wind chill ini.SetValue("WindChill", "Low", HiLoYest.LowWindChill); - ini.SetValue("WindChill", "LTime", HiLoYest.LowWindChillTime.ToString("HH:mm")); + ini.SetValue("WindChill", "LTime", HiLoYest.LowWindChillTime); // Dewpoint ini.SetValue("Dewpoint", "Low", HiLoYest.LowDewPoint); - ini.SetValue("Dewpoint", "LTime", HiLoYest.LowDewPointTime.ToString("HH:mm")); + ini.SetValue("Dewpoint", "LTime", HiLoYest.LowDewPointTime); ini.SetValue("Dewpoint", "High", HiLoYest.HighDewPoint); - ini.SetValue("Dewpoint", "HTime", HiLoYest.HighDewPointTime.ToString("HH:mm")); + ini.SetValue("Dewpoint", "HTime", HiLoYest.HighDewPointTime); // Solar ini.SetValue("Solar", "HighSolarRad", HiLoYest.HighSolar); - ini.SetValue("Solar", "HighSolarRadTime", HiLoYest.HighSolarTime.ToString("HH:mm")); + ini.SetValue("Solar", "HighSolarRadTime", HiLoYest.HighSolarTime); ini.SetValue("Solar", "HighUV", HiLoYest.HighUv); - ini.SetValue("Solar", "HighUVTime", HiLoYest.HighUvTime.ToString("HH:mm")); + ini.SetValue("Solar", "HighUVTime", HiLoYest.HighUvTime); // Feels like ini.SetValue("FeelsLike", "Low", HiLoYest.LowFeelsLike); - ini.SetValue("FeelsLike", "LTime", HiLoYest.LowFeelsLikeTime.ToString("HH:mm")); + ini.SetValue("FeelsLike", "LTime", HiLoYest.LowFeelsLikeTime); ini.SetValue("FeelsLike", "High", HiLoYest.HighFeelsLike); - ini.SetValue("FeelsLike", "HTime", HiLoYest.HighFeelsLikeTime.ToString("HH:mm")); + ini.SetValue("FeelsLike", "HTime", HiLoYest.HighFeelsLikeTime); // Humidex ini.SetValue("Humidex", "High", HiLoYest.HighHumidex); - ini.SetValue("Humidex", "HTime", HiLoYest.HighHumidexTime.ToString("HH:mm")); + ini.SetValue("Humidex", "HTime", HiLoYest.HighHumidexTime); ini.Flush(); diff --git a/Updates.txt b/Updates.txt index b8c5c7c0..cefeeeb4 100644 --- a/Updates.txt +++ b/Updates.txt @@ -7,7 +7,7 @@ BETA 4037 Changes: - Query Daily Data to date now inclusive - Daily data for query now added to at rollover, and amended at day file edits - Fix Davis Cloud Station continually attempting to download history data on error - +- Fix a lurking problem with the today.ini and yesterday.ini files that has been there from day 0 BETA 4036 Changes: —————————————————— @@ -95,6 +95,7 @@ Fixed - Example if you have a main station Davis with sensors 1 & 2, their units will be cb, and you have Ecowitt extra sensors 3 & 4, their units will be % - This does mean a change to the Units JSON and the graph scripts for the default web site. A re-upload of /js/cumuluscharts.js and /js/selectachart.js will be required - Fix Davis Cloud Station continually attempting to download history data on error +- Fix a lurking problem with the today.ini and yesterday.ini files that has been there from day 0. Times are now stored as a full date/time Package Updates From 8de91d30a59d56e5e6ae379f856b01abeecde896 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sun, 22 Sep 2024 16:34:50 +0100 Subject: [PATCH 56/63] Fix crash when stopping as a service - Console.CursorVisible = true --- CumulusMX/CumulusMX.csproj | 2 +- CumulusMX/Program.cs | 23 ++++++++++++++++++----- 2 files changed, 19 insertions(+), 6 deletions(-) diff --git a/CumulusMX/CumulusMX.csproj b/CumulusMX/CumulusMX.csproj index c694d44e..4b5a0e99 100644 --- a/CumulusMX/CumulusMX.csproj +++ b/CumulusMX/CumulusMX.csproj @@ -55,7 +55,7 @@ - 4.2.0.4037-beta + 4.2.0.4038 Copyright © 2015-$([System.DateTime]::Now.ToString('yyyy')) Cumulus MX $(AssemblyName) diff --git a/CumulusMX/Program.cs b/CumulusMX/Program.cs index 0ac65ec4..df00c0cb 100644 --- a/CumulusMX/Program.cs +++ b/CumulusMX/Program.cs @@ -94,7 +94,10 @@ private static async Task Main(string[] args) svcTextListener.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff ") + "Cumulus terminating"); } - Console.CursorVisible = true; + if (!service) + { + Console.CursorVisible = true; + } }; @@ -116,7 +119,10 @@ private static async Task Main(string[] args) Trace.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff ") + "Cumulus has shutdown"); ev.Cancel = true; exitSystem = true; - Console.CursorVisible = true; + if (!service) + { + Console.CursorVisible = true; + } }; AppDomain.CurrentDomain.UnhandledException += UnhandledExceptionTrapper; @@ -306,7 +312,10 @@ private static async Task Main(string[] args) await Task.Delay(500); } - Console.CursorVisible = true; + if (!service) + { + Console.CursorVisible = true; + } } private static void Usage() @@ -321,7 +330,11 @@ private static void Usage() Console.WriteLine(" -user - Specifies the user to run the service under (Linux only)"); Console.WriteLine(" -service - Must be used when running as service (Linux only)"); Console.WriteLine("\nCumulus terminating"); - Console.CursorVisible = true; + try + { + Console.CursorVisible = true; + } + catch { } Environment.Exit(1); } @@ -342,7 +355,6 @@ private static void UnhandledExceptionTrapper(object sender, UnhandledExceptionE { try { - Console.CursorVisible = true; Trace.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff ") + "!!! Unhandled Exception !!!"); Trace.WriteLine(e.ExceptionObject.ToString()); @@ -354,6 +366,7 @@ private static void UnhandledExceptionTrapper(object sender, UnhandledExceptionE } else { + Console.CursorVisible = true; Console.WriteLine(e.ExceptionObject.ToString()); Console.WriteLine("**** An error has occurred - please zip up the MXdiags folder and post it in the forum ****"); Console.WriteLine("Press Enter to terminate"); From f6c4f620fe85232a9ba0338f0dd6b2ad546cf03a Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Sun, 22 Sep 2024 16:55:45 +0100 Subject: [PATCH 57/63] Updates.txt --- Updates.txt | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/Updates.txt b/Updates.txt index cefeeeb4..c50638b2 100644 --- a/Updates.txt +++ b/Updates.txt @@ -1,5 +1,9 @@ -4.2.0 - b4037 +4.2.0 - b4038 ————————————— +BETA 4038 Changes: +—————————————————— +- Fix crash when stopping service + BETA 4037 Changes: —————————————————— - Configuration Wizard now working again From 89e9f05274763a075e3c936b8a546bc918fff456 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Tue, 24 Sep 2024 12:18:04 +0100 Subject: [PATCH 58/63] Fix crash in getting Ecowitt firmware versions --- CumulusMX/EcowittApi.cs | 1 + CumulusMX/EcowittCloudStation.cs | 64 +++++++++++++++++++------------- CumulusMX/HttpStationEcowitt.cs | 45 ++++++++++++++++------ Updates.txt | 4 +- 4 files changed, 76 insertions(+), 38 deletions(-) diff --git a/CumulusMX/EcowittApi.cs b/CumulusMX/EcowittApi.cs index fc6647b4..3f767836 100644 --- a/CumulusMX/EcowittApi.cs +++ b/CumulusMX/EcowittApi.cs @@ -2504,6 +2504,7 @@ internal string[] GetStationList(bool CheckCamera, string macAddress, Cancellati // weather station - check the version vers = stn.stationtype.Split('V')[^1]; model = stn.stationtype.Replace("_", string.Empty).Split('V')[0]; + break; } else { diff --git a/CumulusMX/EcowittCloudStation.cs b/CumulusMX/EcowittCloudStation.cs index b6c93cf7..26ae6f08 100644 --- a/CumulusMX/EcowittCloudStation.cs +++ b/CumulusMX/EcowittCloudStation.cs @@ -106,30 +106,37 @@ public EcowittCloudStation(Cumulus cumulus, WeatherStation station = null) : bas ecowittApi = new EcowittApi(cumulus, this); // Only perform the Start-up if we are a proper station, not a Extra Sensor - if (mainStation) + try { - Task.Run(getAndProcessHistoryData); - var retVal = ecowittApi.GetStationList(true, cumulus.EcowittMacAddress, cumulus.cancellationToken); - if (retVal.Length == 2 && !retVal[1].StartsWith("EasyWeather")) + if (mainStation) { - // EasyWeather seems to contain the WiFi version - deviceFirmware = new Version(retVal[0]); - deviceModel = retVal[1]; + Task.Run(getAndProcessHistoryData); + var retVal = ecowittApi.GetStationList(true, cumulus.EcowittMacAddress, cumulus.cancellationToken); + if (retVal.Length == 2 && !retVal[1].StartsWith("EasyWeather")) + { + // EasyWeather seems to contain the WiFi version + deviceFirmware = new Version(retVal[0]); + deviceModel = retVal[1]; + } } - } - else - { - // see if we have a camera attached - var retVal = ecowittApi.GetStationList(cumulus.EcowittExtraUseCamera, cumulus.EcowittMacAddress, cumulus.cancellationToken); - if (retVal.Length == 2 && !retVal[1].StartsWith("EasyWeather")) + else { - // EasyWeather seems to contain the WiFi version - deviceFirmware = new Version(retVal[0]); - deviceModel = retVal[1]; + // see if we have a camera attached + var retVal = ecowittApi.GetStationList(cumulus.EcowittExtraUseCamera, cumulus.EcowittMacAddress, cumulus.cancellationToken); + if (retVal.Length == 2 && !retVal[1].StartsWith("EasyWeather")) + { + // EasyWeather seems to contain the WiFi version + deviceFirmware = new Version(retVal[0]); + deviceModel = retVal[1]; + } } - } - _ = CheckAvailableFirmware(); + _ = CheckAvailableFirmware(); + } + catch (Exception ex) + { + cumulus.LogExceptionMessage(ex, "Error checking firmware version"); + } } public override void Start() @@ -178,15 +185,22 @@ public override void Start() if (hour == 13) { - var retVal = ecowittApi.GetStationList(mainStation || cumulus.EcowittExtraUseCamera, cumulus.EcowittMacAddress, cumulus.cancellationToken); - if (retVal.Length == 2 && !retVal[1].StartsWith("EasyWeather")) + try + { + var retVal = ecowittApi.GetStationList(mainStation || cumulus.EcowittExtraUseCamera, cumulus.EcowittMacAddress, cumulus.cancellationToken); + if (retVal.Length == 2 && !retVal[1].StartsWith("EasyWeather")) + { + // EasyWeather seems to contain the WiFi version + deviceFirmware = new Version(retVal[0]); + deviceModel = retVal[1]; + GW1000FirmwareVersion = retVal[0]; + } + _ = CheckAvailableFirmware(); + } + catch (Exception ex) { - // EasyWeather seems to contain the WiFi version - deviceFirmware = new Version(retVal[0]); - deviceModel = retVal[1]; - GW1000FirmwareVersion = retVal[0]; + cumulus.LogExceptionMessage(ex, "Error decoding firmware version"); } - _ = CheckAvailableFirmware(); } } } diff --git a/CumulusMX/HttpStationEcowitt.cs b/CumulusMX/HttpStationEcowitt.cs index a177b57c..dfd11a1e 100644 --- a/CumulusMX/HttpStationEcowitt.cs +++ b/CumulusMX/HttpStationEcowitt.cs @@ -138,11 +138,18 @@ public HttpStationEcowitt(Cumulus cumulus, WeatherStation station = null) : base } else { - var retVal = ecowittApi.GetStationList(true, cumulus.EcowittMacAddress, cumulus.cancellationToken); - if (retVal.Length == 2 && !retVal[1].StartsWith("EasyWeather") && !string.IsNullOrEmpty(retVal[0])) + try + { + var retVal = ecowittApi.GetStationList(true, cumulus.EcowittMacAddress, cumulus.cancellationToken); + if (retVal.Length == 2 && !retVal[1].StartsWith("EasyWeather") && !string.IsNullOrEmpty(retVal[0])) + { + deviceFirmware = new Version(retVal[0]); + deviceModel = retVal[1]; + } + } + catch (Exception ex) { - deviceFirmware = new Version(retVal[0]); - deviceModel = retVal[1]; + cumulus.LogExceptionMessage(ex, "Error getting station firmware version"); } } } @@ -155,11 +162,18 @@ public HttpStationEcowitt(Cumulus cumulus, WeatherStation station = null) : base } else { - var retVal = ecowittApi.GetStationList(cumulus.EcowittExtraUseCamera, cumulus.EcowittMacAddress, cumulus.cancellationToken); - if (retVal.Length == 2 && !retVal[1].StartsWith("EasyWeather") && !string.IsNullOrEmpty(retVal[0])) + try { - deviceFirmware = new Version(retVal[0]); - deviceModel = retVal[1]; + var retVal = ecowittApi.GetStationList(cumulus.EcowittExtraUseCamera, cumulus.EcowittMacAddress, cumulus.cancellationToken); + if (retVal.Length == 2 && !retVal[1].StartsWith("EasyWeather") && !string.IsNullOrEmpty(retVal[0])) + { + deviceFirmware = new Version(retVal[0]); + deviceModel = retVal[1]; + } + } + catch(Exception ex) + { + cumulus.LogExceptionMessage(ex, "Error getting camera firmware version"); } } cumulus.LogMessage("Extra Sensors - HTTP Station (Ecowitt) - Waiting for data..."); @@ -1095,11 +1109,18 @@ public string ApplyData(string dataString, bool main, DateTime? ts = null) } else { - var retVal = ecowittApi.GetStationList(main || cumulus.EcowittExtraUseCamera, cumulus.EcowittMacAddress, cumulus.cancellationToken); - if (retVal.Length == 2 && !retVal[1].StartsWith("EasyWeather")) + try { - deviceFirmware = new Version(retVal[0]); - deviceModel = retVal[1]; + var retVal = ecowittApi.GetStationList(main || cumulus.EcowittExtraUseCamera, cumulus.EcowittMacAddress, cumulus.cancellationToken); + if (retVal.Length == 2 && !retVal[1].StartsWith("EasyWeather")) + { + deviceFirmware = new Version(retVal[0]); + deviceModel = retVal[1]; + } + } + catch (Exception ex) + { + cumulus.LogExceptionMessage(ex, "Error getting firmware version"); } } diff --git a/Updates.txt b/Updates.txt index c50638b2..e49c6134 100644 --- a/Updates.txt +++ b/Updates.txt @@ -3,6 +3,7 @@ BETA 4038 Changes: —————————————————— - Fix crash when stopping service +- Fix crash in Ecowitt getting station firmware version in some circumstances BETA 4037 Changes: —————————————————— @@ -61,7 +62,7 @@ BETA 4033 Changes: New - New station type: Ecowitt Local HTTP API - Ecowitt Local HTTP API is an alternative the local TCP API used by the gateways and some stations - - Currently it is less capable than the TCP API + - Currently it is slightly less capable than the TCP API, but does provide all the sensor values - Allows direct support of Ecowitt stations that do not support the TCP API and currently have to use the Custom HTTP Server mode - Exposes the UseDataLogger setting in Station Settings > Options > Advanced - Implements a new data viewer where you can select and view historic data from the monthly logs for a given period, for a set of data values. See: "Data logs > Interval Data Viewer" @@ -100,6 +101,7 @@ Fixed - This does mean a change to the Units JSON and the graph scripts for the default web site. A re-upload of /js/cumuluscharts.js and /js/selectachart.js will be required - Fix Davis Cloud Station continually attempting to download history data on error - Fix a lurking problem with the today.ini and yesterday.ini files that has been there from day 0. Times are now stored as a full date/time +- Fix crash in Ecowitt getting station firmware version in some circumstances Package Updates From b9c74b669c019fa1ef79b1742ed3141808306e6d Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Tue, 24 Sep 2024 14:42:49 +0100 Subject: [PATCH 59/63] Anoer tweek to Ecowitt firmware versions --- CumulusMX/EcowittApi.cs | 13 +++++++------ 1 file changed, 7 insertions(+), 6 deletions(-) diff --git a/CumulusMX/EcowittApi.cs b/CumulusMX/EcowittApi.cs index 3f767836..ed0410ba 100644 --- a/CumulusMX/EcowittApi.cs +++ b/CumulusMX/EcowittApi.cs @@ -2498,25 +2498,26 @@ internal string[] GetStationList(bool CheckCamera, string macAddress, Cancellati { // we have a camera cumulus.EcowittCameraMacAddress = stn.mac; + cumulus.LogDebugMessage($"API.GetStationList: Found Camera name={stn.name ?? "-"}, vers={stn.stationtype ?? "-"}"); } else if (stn.type == 1 && stn.mac.Equals(macAddress, StringComparison.CurrentCultureIgnoreCase)) { // weather station - check the version vers = stn.stationtype.Split('V')[^1]; model = stn.stationtype.Replace("_", string.Empty).Split('V')[0]; - break; + cumulus.LogDebugMessage($"API.GetStationList: Found Station model={model}, vers={vers}"); } else { // no idea what we got! - vers = stn.name; - model = stn.name; + cumulus.LogDebugMessage($"API.GetStationList: Found type={stn.type}, name={stn.name?? "-"}, model={stn.stationtype ?? "-"}"); } } - cumulus.LogDebugMessage($"API.GetStationList: Found vers={vers}, model={model}"); - - return [vers, model]; + if (vers != string.Empty && model != string.Empty) + { + return [vers, model]; + } } return []; From 91bb8ea371476de624f672ad99b91cab2fe1e6dd Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Mon, 30 Sep 2024 15:08:08 +0100 Subject: [PATCH 60/63] add dashed value for console battery web tag --- CumulusMX/CumulusMX.csproj | 4 +- CumulusMX/DavisAirLink.cs | 2 +- CumulusMX/HttpStationAmbient.cs | 6 +-- CumulusMX/Properties/launchSettings.json | 2 +- CumulusMX/webtags.cs | 4 +- Updates.txt | 61 +----------------------- 6 files changed, 9 insertions(+), 70 deletions(-) diff --git a/CumulusMX/CumulusMX.csproj b/CumulusMX/CumulusMX.csproj index 4b5a0e99..fa4c4124 100644 --- a/CumulusMX/CumulusMX.csproj +++ b/CumulusMX/CumulusMX.csproj @@ -55,7 +55,7 @@ - 4.2.0.4038 + 4.2.0.4039 Copyright © 2015-$([System.DateTime]::Now.ToString('yyyy')) Cumulus MX $(AssemblyName) @@ -128,7 +128,7 @@ - + \ No newline at end of file diff --git a/CumulusMX/DavisAirLink.cs b/CumulusMX/DavisAirLink.cs index 8eae2db2..b9c0e80f 100644 --- a/CumulusMX/DavisAirLink.cs +++ b/CumulusMX/DavisAirLink.cs @@ -316,7 +316,7 @@ private async void GetAlCurrent(object source, ElapsedEventArgs e) var urlCurrent = $"http://{ip}/v1/current_conditions"; - WebReq.Wait(); + await WebReq.WaitAsync(); // The AL will error if already responding to a request from another device, so add a retry do diff --git a/CumulusMX/HttpStationAmbient.cs b/CumulusMX/HttpStationAmbient.cs index 109fc74a..0294ff9d 100644 --- a/CumulusMX/HttpStationAmbient.cs +++ b/CumulusMX/HttpStationAmbient.cs @@ -10,14 +10,12 @@ namespace CumulusMX class HttpStationAmbient : WeatherStation { private readonly WeatherStation station; - private readonly Cumulus cumulus; private bool starting = true; private bool stopping = false; public HttpStationAmbient(Cumulus cumulus, WeatherStation station = null) : base(cumulus, station != null) { this.station = station; - this.cumulus = cumulus; if (station == null) { @@ -775,14 +773,14 @@ private void ProcessLightning(NameValueCollection data, WeatherStation station) { // Only set the lightning time/distance if it is newer than what we already have - the GW1000 seems to reset this value var valDist = Convert.ToDouble(dist, CultureInfo.InvariantCulture); - if (valDist != 255) + if (valDist < 255) { LightningDistance = ConvertUnits.KmtoUserUnits(valDist); } var valTime = Convert.ToDouble(time, CultureInfo.InvariantCulture); // Sends a default value until the first strike is detected of 0xFFFFFFFF - if (valTime != 0xFFFFFFFF) + if (valTime < 0xFFFFFFFF) { var dtDateTime = DateTime.UnixEpoch; dtDateTime = dtDateTime.AddSeconds(valTime).ToLocalTime(); diff --git a/CumulusMX/Properties/launchSettings.json b/CumulusMX/Properties/launchSettings.json index e4a0e3dc..781e7aac 100644 --- a/CumulusMX/Properties/launchSettings.json +++ b/CumulusMX/Properties/launchSettings.json @@ -2,7 +2,7 @@ "profiles": { "CumulusMX": { "commandName": "Project", - "commandLineArgs": "-lang en-GB" + "commandLineArgs": "-lang en-GB -port 8998" } } } \ No newline at end of file diff --git a/CumulusMX/webtags.cs b/CumulusMX/webtags.cs index 8f552b46..f2361ff3 100644 --- a/CumulusMX/webtags.cs +++ b/CumulusMX/webtags.cs @@ -1034,12 +1034,12 @@ private string Tagintemp(Dictionary tagParams) private string Tagbattery(Dictionary tagParams) { - return CheckRc(station.ConBatText, tagParams); + return CheckRc(station.ConBatText ?? "--", tagParams); } private string TagConsoleSupplyV(Dictionary tagParams) { - return CheckRc(station.ConSupplyVoltageText, tagParams); + return CheckRc(station.ConSupplyVoltageText ?? "--", tagParams); } private string Tagtxbattery(Dictionary tagParams) diff --git a/Updates.txt b/Updates.txt index e49c6134..c06a4012 100644 --- a/Updates.txt +++ b/Updates.txt @@ -1,64 +1,5 @@ -4.2.0 - b4038 +4.2.0 - b4039 ————————————— -BETA 4038 Changes: -—————————————————— -- Fix crash when stopping service -- Fix crash in Ecowitt getting station firmware version in some circumstances - -BETA 4037 Changes: -—————————————————— -- Configuration Wizard now working again -- Station Settings for HTTP API station now drops MAC to match Wizard settings -- Query Daily Data to date now inclusive -- Daily data for query now added to at rollover, and amended at day file edits -- Fix Davis Cloud Station continually attempting to download history data on error -- Fix a lurking problem with the today.ini and yesterday.ini files that has been there from day 0 - -BETA 4036 Changes: -—————————————————— -- Accessibility updates for the Interval and Daily data viewer pages -- Add Soil moisture sensors 9-16 to Ecowitt HTTP API decode -- Add Soil moisture sensors 9-16 to Ecowitt TCP API decode of sensor info -- Soil moisture units now follow the source -- Two new web tags <#NewRecordAlarm> and <#NewRecordAlarmMessage> -- Delete old MD5 hash files -- New day file query web tag <#DayFileQuery>. Please read the separate documentation (QueryDayFileWebTag.md) for more details -- Fix Ecowitt HTTP API station not showing in the config Wizard -- Add bash script for old Fine Offset USB stations on Linux - - -BETA 4035 Changes: -—————————————————— -- Fixes to the Interval/Daily data viewer web pages -- Fix Ecowitt HTTP API station parsing of some decimal values when running in a user locale that uses comma decimals -- Fix Ecowitt HTTP API station settings not having a field for the MAC address -- Add support for Ecowitt srain_piezo to Ecowitt HTTP API, and HTTP Custom Sender stations - - -BETA 4034 Changes: -—————————————————— -- Improved Ecowitt HTTP API station error handling/recovery when station goes offline -- Daily and Interval data viewers now visible on the dashboard -- Removed Ecowitt HTTP API station password -- New web tag <#MonthRainfall> - - -BETA 4033 Changes: -—————————————————— -- Add Ecowitt HTTP API password -- Add polling time update to Ecowitt HTTP API station -- Add more Ecowitt HTTP API station error handling -- Ecowitt HTTP API station ignore apparent temp -- More HTTP API station Lightning fixes -- Davis WL Cloud station - some support for new <#LowBattList> -- Davis WLL station support for new <#LowBattList> -- Davis VP2 station support for new <#LowBattList> -- Detect 404 errors for the Ecowitt HTTP API -- External Programs now set the working directory to the location of the executable/script rather than the Cumulus MX home directory -- Allow for Ecowitt Lightning sensor factory reset/new device -- Latest AI2 updates applied - - New - New station type: Ecowitt Local HTTP API - Ecowitt Local HTTP API is an alternative the local TCP API used by the gateways and some stations From b5e8f1d8f80fd560d483754a1fe0d6f8a9a6e42c Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Mon, 30 Sep 2024 23:36:42 +0100 Subject: [PATCH 61/63] Updates.txt --- Updates.txt | 3 +++ 1 file changed, 3 insertions(+) diff --git a/Updates.txt b/Updates.txt index c06a4012..ee961ddb 100644 --- a/Updates.txt +++ b/Updates.txt @@ -48,6 +48,9 @@ Fixed Package Updates - SQLite: Reverted to v2.1.8 pending fix from author - MQTTnet +- FluentFTP +- ServiceStack.Text +- SixLabors.ImageSharp 4.1.3 - b4028 From e4204117ebae4e999b5615e2c2fc3b4670e3c7f0 Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Tue, 1 Oct 2024 15:50:23 +0100 Subject: [PATCH 62/63] Change Updates.txt to CHANGELOG.md --- .gitignore | 1 - CHANGELOG.md | 274 +++ CumulusMX/CumulusMX.csproj | 2 +- README.md | 13 +- Updates.txt | 3229 ------------------------------------ 5 files changed, 283 insertions(+), 3236 deletions(-) create mode 100644 CHANGELOG.md delete mode 100644 Updates.txt diff --git a/.gitignore b/.gitignore index 62535d7a..e1ece7af 100644 --- a/.gitignore +++ b/.gitignore @@ -189,4 +189,3 @@ FakesAssemblies/ ToDo.txt /CumulusMX/DataEditor-byDay.cs /CumulusMX/WeatherStation-byDay.cs -/Updates.md diff --git a/CHANGELOG.md b/CHANGELOG.md new file mode 100644 index 00000000..b207a132 --- /dev/null +++ b/CHANGELOG.md @@ -0,0 +1,274 @@ +# Changelog + +All notable changes to this project will be documented in this file. + +Additional notes are available on the [forum release thread](https://cumulus.hosiene.co.uk/viewtopic.php?t=17887) + +This file is formatted as [markdown](https://www.markdownguide.org/), any decent editor should display it correctly formatted. + +--- + +## 4.2.0 \[b4039\] - 2024-10-01 + +### New +- New station type: Ecowitt Local HTTP API + - Ecowitt Local HTTP API is an alternative the local TCP API used by the gateways and some stations + - Currently it is slightly less capable than the TCP API, but does provide all the sensor values + - Allows direct support of Ecowitt stations that do not support the TCP API and currently have to use the Custom HTTP Server mode +- Exposes the UseDataLogger setting in `Station Settings > Options > Advanced` +- Implements a new data viewer where you can select and view historic data from the monthly logs for a given period, for a set of data values. See: `Data logs > Interval Data Viewer` +- Implements a new data viewer where you can select and view historic daily data from the day file for a given period, for a set of data values. See: `Data logs > Daily Data Viewer` +- New web tag `<#LowBatteryList>` which returns the list of sensors/transmitters that have low batteries + - The format is a comma separated list of sensor/transmitter IDs and battery states + - Eg. "wh80-<state>,wh41ch1-<state>,wh41ch2-<state>" + - Where `` is "LOW" or "0" or "1" depending on what the sensor sends +- New web tag `<#MonthRainfall>` which returns the rainfall total for the current month by default + - Takes optional parameters `y=YYYY` and `m=MM` (both must be specified) to return the total rainfall for specified month in the specified year + - Eg. `<#MonthRainfall y=2018 m=10>` +- Support for Ecowitt WS90 piezo IsRaining status to trigger MX IsRaining. + - Currently only supported with a WS90/WS85 connected to a GW2000 (Sept. 2024). This value is being added to more stations as they get firmware updates. +- Two new web tags `<#NewRecordAlarm>` and `<#NewRecordAlarmMessage>` + - NewRecordAlarm somewhat replicates the existing #newrecord web tag, but is also controlled by the alarm being enable/disabled + - NewRecordAlarmMessage displays the last new record alarm text message +- Old MD5 hash files are now deleted on startup +- New web tag `<#DayFileQuery>` which allows flexible querying of the day file. + - Please read the separate documention (`/MXutils/QueryDayFile.md`) for more details +- Added a script `/MXutils/linux/Fix_FineOffset_USB.sh` to fix Fine Offset USB stations + +### Changed +- Davis WLL/Davis Cloud stations now use the WL.com subscription level to determine if they use WL.com as a logger +- The Station Settings screen now does a two-stage selection: First the manufacturer, then the station model. This shortens the long and list to select from. +- AI2 dashboard now shows some Davis WLL hardware information +- External Programs now sets the working directory to the location of the executable/script rather than the Cumulus MX home directory +- Latest AI2 updates applied +- Updates.txt is now `CHANGELOG.md` + +### Fixed +- Davis Cloud stations in endless loop at startup if there is no historic data to process or access is denied +- Davis Cloud stations no longer continuously try to fetch history data if there is no Pro subscription +- WMR928 Station now correctly converts indoor temperatures to the user defined units +- Not logging PHP upload failures to the warning log +- Soil moisture units now follow the source + - Example if you have a main station Davis with sensors 1 & 2, their units will be cb, and you have Ecowitt extra sensors 3 & 4, their units will be % + - This does mean a change to the Units JSON and the graph scripts for the default web site. A re-upload of `/js/cumuluscharts.js` and `/js/selectachart.js` will be required +- Fix Davis Cloud Station continually attempting to download history data on error +- Fix a lurking problem with the `today.ini` and `yesterday.ini` files that has been there from day 0. Times are now stored as a full date/time +- Fix crash in Ecowitt getting station firmware version in some circumstances + +### Package Updates +- SQLite: Reverted to v2.1.8 pending fix from author +- MQTTnet +- FluentFTP +- ServiceStack.Text +- SixLabors.ImageSharp + +--- + +## 4.1.3 \[b4028\] - 2024-08-20 + +### New +- New web tag `<#stationId>` which returns the internal station number used by CMX to determine the station type +- For Davis WLL and WeatherLink Cloud stations you can now specify the station identifier using the stations UUID instead of the numeric Id. The UUID is simpler to find + as it forms part of the URL of every web page related to your station on weatherlink.com + +### Changed +No changes + +### Fixed +- The Cumulus MX version comparison with latest online at startup and daily +- Fix CMX version check when no betas are available on GitHub repo +- Davis Cloud Station can now accurately determine the current conditions update rate +- Fix Davis WLL (and others) creating erroneous wind speed spike warnings +- Alternative Interface 2 - Davis reception stats display incorrectly +- Davis Cloud Station (VP2) now correctly displays the Davis ET values when "Cumulus calculates ET" is not enabled + - Note: If "Cumulus calculates ET" is not enabled, the last hours ET every day, will be accumulated in the first hour of the following day +- Davis WLL, and Davis Cloud stations, fixed a problem where the rollover would not be performed if historic data was not available and MX was stopped before the rollover and restarted after +- Improved Ctrl-C shutdown of Cumulus MX for Davis VP2 stations when they are failing to connect with the station +- Fix Ecowitt firmware check when running test firmware + +### Package Updates +- SixLabors.ImageSharp +- FluentFTP +- MailKit +- NReco.Logging.File +- ServiceStack.Text +- SSH.Net +- SQLite + +--- + +## 4.1.2 \[b4027\] - 2024-07-23 + +### New +- Adds wind run to the dashboard "now" page +- Adds support for the format parameter to the `<#ProgramUpTime>` and `<#SystemUpTime>` web tags + - The format syntax is different from date/time web tags as these two tags use a elapsed time. + - For Custom format specifiers see: [Custom TimeSpan formats](https://learn.microsoft.com/en-us/dotnet/standard/base-types/custom-timespan-format-strings) + - For Standard format specifiers see: [Standard TimeSpan formats](https://learn.microsoft.com/en-us/dotnet/standard/base-types/standard-timespan-format-strings) + - The default output is generated using the format string `"{0:%d} days {0:%h} hours"` + - You can customise this like this example: `<#SystemUpTime format="{0:%d}d {0:%h}h {0:%m}m">` --> "12d 9h 46m" +- New web tag `<#AnnualRainfall>` + - Defaults to the current year if no year is specified, so is equivalent to the preferred web tag `<#ryear>` + - Accepts a tag parameter of `y=nnnn`, which will return the total rainfall for the specified year. Eg. `<#AnnualRainfall y=2021>` + +### Changed +- Daily backup now runs asynchronously to prevent it stopping MX continue to run + +### Fixed +- Davis VP2 connection type being decoded from Cumulus.ini incorrectly +- Add missing knots input to JSON station +- Solar W/m2 should use superscript in dashboard +- Cumulus Calculates SLP giving a spike on start-up with the following stations: Ecowitt Local API, Davis Cloud Station, Ecowitt Cloud Station, HTTP Ambient, HTTP Ecowitt +- Add missing wind run units and extra space in humidity % in ai2 dashboard +- Remove UV/Solar missing data messages from Davis Cloud (VP2) +- A new version of ***MigrateData3to4*** (1.0.3) to fix issues migrating the day file +- Negative 0.0 appearing when no rainfall has occurred +- Davis Cloud Station wind processing changed, instead of MX trying to calculate an average wind speed, MX now uses the wind speed data from Davis directly.
+ The data granularity from the cloud is not enough for the average to be calculated. + +--- + +## 4.1.1 \[b4025\] - 2024-06-19 + +### New +No new features + +### Changed +No changes + +### Fixed +- Davis VP2/Vue raincounter reset problems +- Another raincounter reset issue that has been lurking +- Wizard made Ecowitt API key and secret mandatory +- Fix for FTP overwrite performing delete + create of remote file + +--- + +## 4.1.0 \[b4024\] - 2024-06-05 + +### New +- HTTP (Ecowitt) station now accepts the data via a simple GET url as well as POST +- Cumulus now calculates the AQi for Ecowitt PM and CO₂ sensors + - New web tags: + + `<#AirQualityIdx1[-4]>`, `<#AirQualityAvgIdx1[-4]>`
+ `<#CO2_pm2p5_aqi>`, `<#CO2_pm2p5_24h_aqi>`
+ `<#CO2_pm10_aqi>`, `<#CO2_pm10_24_aqi>` + +- Add new pressure units option of kilopascal (kPa) +- New station type added: JSON Data Input, marked as "experimental" for now, but testing so far has been successful + - Accepts data in a JSON format defined in `MXutils/WeatherStationInput.jsonc` + - Input mechanism is via: + - Named file + - HTTP POST to `http://[CMX_IP_Address]:8998/station/json` + - MQTT using a named topic +- Locale Strings now has settings for the default record date/time text + +### Changed +- Removed option for WOW catch-up, it isn't supported by WOW +- Moved the log file header info files to the `MXutils/fileheaders` folder + +### Fixed +- Temperature Sum graph data when Sum0 is the only selected range +- Fix `<#NewBuildAvailable>` and `<#NewBuildNumber>` web tags +- Fix for Davis VP2 consoles losing todays rainfall on a full power cycle +- Exception when enabling real-time FTP whilst running and FTP logging is enabled +- Davis WLL now fires a single "sensor contact lost" warning message + contact restored +- Fix for multiple realtime FTP log-ins being attempted in parallel +- Alarm actions errored if the action parameter field is empty + +### Package Updates +- MQTTnet +- MailKit +- BouncyCastle + +--- + +## 4.0.1 \[b4023\] - 2024-05-16 + +### New +- There is now a 32 Windows specific version of executable - `CumulusMX32.exe` + - The same applies to `MigrateData3to4` and `CreateMissing` + +### Changed +- Removed the experimental Gmail OATH2 authentication method +- Third party uploads now have retries and the timeout increased to 30 seconds + + +### Fixed +- Fixed Spike handling for outdoor temperature +- Fixed David Cloud (VP2) station sometimes not decoding dew point, adds indoor temp/hum decode +- The -install option now works on 32 bit Windows + +--- + +## 4.0.0 \[b4022\] - 2024-05-11 + +Initial release of Cumulus MX which now runs under Microsoft .NET 8.0 and removes the requirement for the Mono runtime environment on Linux. + +### New +- Moon Image now supports transparent shadows +- The -install/-unistall command line switches now support both Windows and Linux + - Under Linux run
+ `sudo dotnet CumulusMX.dll -install -user [-port ] [-lang ]` + - Windows install-as-a-service now self-elevates and requests UAC +- Implements encryption of the credentials in the cumulus.ini file + - This requires a new file in the root folder called `UniqueId.txt` + - You **must** copy this `UniqueId.txt` file when you copy the `cumulus.ini` file to new installs, otherwise your sensitive information will not decrypt and you will have to enter it again +- Experimental Gmail OATH 2.0 authentication +- New web tag for the average temperature of the previous 24 hours from now: `<#TempAvg24Hrs>` +- Cumulus backups are now zipped +- Add Enable option to Extra Web Files so you can now save entries but not have them active +- Ecowitt - added firmware update check on start-up and once a day at 13:00 + - New Firmware Alarm to support this + - New web tag `<#FirmwareAlarm>` +- Adds new web tags for temperature means
+ `<#ByMonthTempAvg mon=[1-12]>`
+ Mean for requested month over the entire history. Omit the `mon` parameter for the current month
+ `<#MonthTempAvg m=[1-12] y=[YYYY]>`
+ Mean for the requested specific month. Omit the parameters for the current month
+ `<#YearTempAvg y=[YYYY]>`
+ Mean for the requested year. Omit the y parameter for the current year +- Add "MX calculates Sea Level Pressure" + - Applies to HTTP Ecowitt, HTTP Ambient, GW1000, Ecowitt Cloud, FO, Davis Cloud WLC stations + - When enabled, the pressure calibration is applied to the raw station pressure + - Check your station pressure (Absolute) calibration! +- Adds true Altimeter Pressure calculation to GW1000, Ecowitt HTTP, Ecowitt Cloud + - Check your station pressure (Absolute) calibration! +- Added localisation of records web tag date/time formats + +### Changed +- Now **requires** Microsoft .Net 8.0 rather than mono to run under Linux and MacOS +- All data files are now written/read as invariant - dayfile, monthly log files, extra log files, AirLink, and custom log files + - NOTE: Custom log files may require the user to alter their configuration to use comma separators and add the `rc=y` parameter to numeric web tags +- Monthly log files now renamed to `[yyyyMM]log.txt` to remove localised month name - and now sortable in the file system! +- Added `MigrateData3to4` utility. + Basic workflow: + - Clean install v4 + - Copy v3 Cumulus.ini to root + - Copy v3 `/data` and `/Reports` folders to v4 install + - Rename the `/data` folder to `/datav3` + - Run `MigrateData3to4` + - Done! +- Removed previously deprecated web tags + `CO2-24h, CO2-pm2p5, CO2-pm2p5-24h, CO2-pm10, CO2-temp, CO2-hum` +- Loading dayfile now continues on error and reports total errors - only the first 20 errors are logged +- You now only set the Ecowitt MAC/IMEI address in one place for the various station types + - In Local API settings for GW1000 type + - In Cloud Access API for Cloud and HTTP station types + +### Fixed +- Problems when using a 9am rollover in the records editors for values from the monthly log files +- Select-a-Period charts not respecting the interval dates: Air Quality, CO₂, Soil Moisture, Leaf Wetness +- Calibration Limits not changing when the user changes units - eg initial install +- Potential fix for corruption at the end of all data log files when shutting down +- Error that the username is not set when sending email to a server that requires no authentication +- Improvement to GW1000 API reconnects +- Improved web socket initial connection to send data immediately on dashboard/now/gauges connection +- Fix for soil moisture conversion from percentage to cb in Weather Cloud uploads +- Reload dayfile can now only be run as a single instance +- Improvements to Davis WLL wind handling when: + - Transitioning from catch-up to live running + - No broadcasts are received +- Davis WLL improved recovery from loss of broadcast messages +- Spike/limit improvements diff --git a/CumulusMX/CumulusMX.csproj b/CumulusMX/CumulusMX.csproj index fa4c4124..7bf5c894 100644 --- a/CumulusMX/CumulusMX.csproj +++ b/CumulusMX/CumulusMX.csproj @@ -128,7 +128,7 @@ - + \ No newline at end of file diff --git a/README.md b/README.md index f67d81f6..2b081056 100644 --- a/README.md +++ b/README.md @@ -1,13 +1,16 @@ -# CumulusMX -**The CumulusMX weather program** +# Cumulus MX +**The Cumulus MX weather program** -This is the source code for the CumuluMX weather program written by Steve Loft. +This is the source code for the Cumulus MX weather program written by Steve Loft. A note from Steve when he released the code: >"*[the code is]* Offered completely without support in the hope that it might be useful. The code is very badly structured due to the 'Frankenstein' way it was cobbled together from various places. Some of it is a machine translation of parts of Cumulus 1." - + In order to function correctly the program needs the supporting distribution files and folders. These can be found in [this repo](https://github.com/cumulusmx/CumulusMX-DistributionFiles) -The orginal source was post build b3043 - it incorporated some changes that were being worked on by Steve before he retired. +The original source was post build b3043 - it incorporated some changes that were being worked on by Steve before he retired. The support forum for this software can [be found here](https://cumulus.hosiene.co.uk/) + + +The change log [is here](blob/master/CHANGELOG.md) diff --git a/Updates.txt b/Updates.txt deleted file mode 100644 index ee961ddb..00000000 --- a/Updates.txt +++ /dev/null @@ -1,3229 +0,0 @@ -4.2.0 - b4039 -————————————— -New -- New station type: Ecowitt Local HTTP API - - Ecowitt Local HTTP API is an alternative the local TCP API used by the gateways and some stations - - Currently it is slightly less capable than the TCP API, but does provide all the sensor values - - Allows direct support of Ecowitt stations that do not support the TCP API and currently have to use the Custom HTTP Server mode -- Exposes the UseDataLogger setting in Station Settings > Options > Advanced -- Implements a new data viewer where you can select and view historic data from the monthly logs for a given period, for a set of data values. See: "Data logs > Interval Data Viewer" -- Implements a new data viewer where you can select and view historic daily data from the day file for a given period, for a set of data values. See: "Data logs > Daily Data Viewer" -- New web tag <#LowBatteryList> which returns the list of sensors/transmitters that have low batteries - - The format is a comma separated list of sensor/transmitter IDs and battery states - - Eg. "wh80 - ,wh41ch1-,wh41ch2-" - - Where is "LOW" or "0" or "1" depending on what the sensor sends -- New web tag <#MonthRainfall> which returns the rainfall total for the current month by default - - Takes optional parameters y=YYYY and m=MM (both must be specified) to return the total rainfall for specified month in the specified year - - Eg. <#MonthRainfall y=2018 m=10> -- Support for Ecowitt WS90 piezo IsRaining status to trigger MX IsRaining. - - Currently only supported with a WS90/WS85 connected to a GW2000 (Sept. 2024). This value is being added to more stations as they get firmware updates. -- Two new web tags <#NewRecordAlarm> and <#NewRecordAlarmMessage> - - NewRecordAlarm somewhat replicates the existing #newrecord web tag, but is also controlled by the alarm being enable/disabled - - NewRecordAlarmMessage displays the last new record alarm text message -- Old MD5 hash files are now deleted on startup -- New web tag <#DayFileQuery> which allows flexible querying of the day file. - - Please read the separate documention (/MXutils/QueryDayFile.md) for more details -- Added a script to /MXutils/linux/Fix_FineOffset_USB.sh to fix Fine Offset USB stations - -Changed -- Davis WLL/Davis Cloud stations now use the WL.com subscription level to determine if they use WL.com as a logger -- The Station Settings screen now does a two-stage selection: First the manufacturer, then the station model. This shortens the long and list to select from. -- AI2 dashboard now shows some Davis WLL hardware information -- External Programs now sets the working directory to the location of the executable/script rather than the Cumulus MX home directory -- Latest AI2 updates applied - -Fixed -- Davis Cloud stations in endless loop at startup if there is no historic data to process or access is denied -- Davis Cloud stations no longer continuously try to fetch history data if there is no Pro subscription -- WMR928 Station now correctly converts indoor temperatures to the user defined units -- Not logging PHP upload failures to the warning log -- Soil moisture units now follow the source - - Example if you have a main station Davis with sensors 1 & 2, their units will be cb, and you have Ecowitt extra sensors 3 & 4, their units will be % - - This does mean a change to the Units JSON and the graph scripts for the default web site. A re-upload of /js/cumuluscharts.js and /js/selectachart.js will be required -- Fix Davis Cloud Station continually attempting to download history data on error -- Fix a lurking problem with the today.ini and yesterday.ini files that has been there from day 0. Times are now stored as a full date/time -- Fix crash in Ecowitt getting station firmware version in some circumstances - - -Package Updates -- SQLite: Reverted to v2.1.8 pending fix from author -- MQTTnet -- FluentFTP -- ServiceStack.Text -- SixLabors.ImageSharp - - -4.1.3 - b4028 -————————————— -New -- New web tag <#stationId> which returns the internal station number used by CMX to determine the station type -- For Davis WLL and WeatherLink Cloud stations you can now specify the station identifier using the stations UUID instead of the numeric Id. The UUID is simpler to find - as it forms part of the URL of every web page related to your station on weatherlink.com - -Changed - -Fixed -- The Cumulus MX version comparison with latest online at startup and daily -- Fix CMX version check when no betas are available on GitHub repo -- Davis Cloud Station can now accurately determine the current conditions update rate -- Fix Davis WLL (and others) creating erroneous wind speed spike warnings -- Alternative Interface 2 - Davis reception stats display incorrectly -- Davis Cloud Station (VP2) now correctly displays the Davis ET values when "Cumulus calculates ET" is not enabled - - Note: If "Cumulus calculates ET" is not enabled, the last hours ET every day, will be accumulated in the first hour of the following day -- Davis WLL, and Davis Cloud stations, fixed a problem where the rollover would not be performed if historic data was not available and MX was stopped before the rollover and restarted after -- Improved Ctrl-C shutdown of Cumulus MX for Davis VP2 stations when they are failing to connect with the station -- Fix Ecowitt firmware check when running test firmware - - -Package Updates -- SixLabors.ImageSharp -- FluentFTP -- MailKit -- NReco.Logging.File -- ServiceStack.Text -- SSH.Net -- SQLite - - - -4.1.2 - b4027 -————————————— -New -- Adds wind run to the dashboard "now" page -- Adds support for the format parameter to the <#ProgramUpTime> and <#SystemUpTime> web tags - - The format syntax is different from date/time web tags as these two tags use a elapsed time. - - For Custom format specifiers see: https://learn.microsoft.com/en-us/dotnet/standard/base-types/custom-timespan-format-strings - - For Standard format specifiers see: https://learn.microsoft.com/en-us/dotnet/standard/base-types/standard-timespan-format-strings - - The default output is generated using the format string "{0:%d} days {0:%h} hours" - - You can customise this like this example: <#SystemUpTime format="{0:%d}d {0:%h}h {0:%m}m"> --> "12d 9h 46m" -- New web tag <#AnnualRainfall> - - Defaults to the current year if no year is specified, so = <#ryear> - - Accepts a tag parameter of y=nnnn, which will return the total rainfall for the specified year. Eg. <#AnnualRainfall y=2021> - -Changed -- Daily backup now runs asynchronously to prevent it stopping MX continue to run - -Fixed -- Davis VP2 connection type being decoded from Cumulus.ini incorrectly -- Add missing knots input to JSON station -- Solar W/m2 should use superscript in dashboard -- Cumulus Calculates SLP giving a spike on start-up with the following stations: Ecowitt Local API, Davis Cloud Station, Ecowitt Cloud Station, HTTP Ambient, HTTP Ecowitt -- Add missing wind run units and extra space in humidity % in ai2 dashboard -- Remove UV/Solar missing data messages from Davis Cloud (VP2) -- A new version of MigrateData3to4 (1.0.3) to fix issues migrating the day file -- Negative 0.0 appearing when no rainfall has occurred -- Davis Cloud Station wind processing changed, instead of MX trying to calculate an average wind speed, MX now uses the wind speed data from Davis directly. - The data rate from the cloud is not fast enough for the average to be calculated. - - - -4.1.1 - b4025 -————————————— -Fixed -- Davis VP2/Vue raincounter reset problems -- Another raincounter reset issue that has been lurking -- Wizard made Ecowitt API key and secret mandatory -- Fix for FTP overwrite performing delete + create of remote file - - - -4.1.0 - b4024 -————————————— -New -- HTTP (Ecowitt) station now accepts the data via a simple GET url as well as POST -- Cumulus now calculates the AQi for Ecowitt PM and CO₂ sensors - - New web tags: - <#AirQualityIdx1[-4]>, <#AirQualityAvgIdx1[-4]> - <#CO2_pm2p5_aqi>, <#CO2_pm2p5_24h_aqi>, <#CO2_pm10_aqi>, <#CO2_pm10_24_aqi> -- Add new pressure units option of kilopascal (kPa) -- New station type added: JSON Data Input, marked as "experimental" for now, but testing so far has been successful - - Accepts data in a JSON format defined in MXutils/WeatherStationInput.jsonc - - Input mechanism is via: - - Named file - - HTTP POST to http://[CMX_IP_Address]:8998/station/json - - MQTT using a named topic -- Locale Strings now has settings for the default record date/time text - - -Changed -- Removed option for WOW catch-up, it isn't supported by WOW -- Moved the log file header info files to the MXutils/fileheaders folder - - -Fixed -- Temperature Sum graph data when Sum0 is the only selected range -- Fix #NewBuildAvailable and #NewBuildNumber web tags -- Fix for Davis VP2 consoles losing todays rainfall on a full power cycle -- Exception when enabling real-time FTP whilst running and FTP logging is enabled -- Davis WLL now fires a single "sensor contact lost" warning message + contact restored -- Fix for multiple realtime FTP log-ins being attempted in parallel -- Alarm actions errored if the action parameter field is empty - - -Package Updates -- MQTTnet -- MailKit -- BouncyCastle - - -4.0.1 - b4023 -————————————— -New -- There is now a 32 Windows specific version of executable - CumulusMX32.exe - - The same applies to MigrateData3to4 and CreateMissing - -Changed -- Removed the experimental Gmail OATH2 authentication method -- Third party uploads now have retries and the timeout increased to 30 seconds - - -Fixed -- Fixed Spike handling for outdoor temperature -- Fixed David Cloud (VP2) station sometimes not decoding dew point, adds indoor temp/hum decode -- The -install option now works on 32 bit Windows - - - -4.0.0 - b4022 -————————————— -Initial release of Cumulus MX which now runs under Microsoft .NET 8.0 and removes the requirement for the Mono runtime environment on Linux - -New -- Moon Image now supports transparent shadows -- The -install/-unistall command line switches now support both Windows and Linux - - Under Linux run > sudo dotnet CumulusMX.dll -install -user [-port ] [-lang ] - - Windows install as a service now self-elevates and requests UAC -- Implements encryption of the credentials in the cumulus.ini file -- Experimental Gmail OATH 2.0 authentication -- New web tag for the average temperature of the previous 24 hours from now: <#TempAvg24Hrs> -- Cumulus backups are now zipped -- Add Enable option to Extra Web Files so you can now save entries but not have them active -- Ecowitt - added firmware update check on start-up and once a day at 13:00 - - New Firmware Alarm to support this - - New web tag <#FirmwareAlarm> -- Adds new web tags for temperature means - <#ByMonthTempAvg mon=[1-12]> Mean for requested month over the entire history. Omit the mon parameter for the current month - <#MonthTempAvg m=[1-12] y=[YYYY]> Mean for the requested specific month. Omit the parameters for the current month - <#YearTempAvg y=[YYYY]> Mean for the requested year. Omit the y parameter for the current year -- Add "MX calculates Sea Level Pressure" - - Applies to HTTP Ecowitt, HTTP Ambient, GW1000, Ecowitt Cloud, FO, Davis Cloud WLC stations - - When enabled, the pressure calibration is applied to the raw station pressure - - Check your station pressure (Absolute) calibration! -- Adds true Altimeter Pressure calculation to GW1000, Ecowitt HTTP, Ecowitt Cloud - - Check your station pressure (Absolute) calibration! -- Added localisation of records web tag date/time formats - - -Changed -- Now requires Microsoft .Net 8.0 rather than mono to run under Linux and MacOS -- All data files are now written/read as invariant - dayfile, monthly log files, extra log files, AirLink, and custom log files - - NOTE: Custom log files may require the user to alter their configuration to use comma separators and add the rc=y parameter to numeric web tags -- Monthly log files now renamed to "[yyyyMM]log.txt" to remove localised month name - and now sortable in the file system! -- Added MigrateData3to4 utility. - Basic workflow: - - Clean install v4 - - Copy v3 Cumulus.ini to root - - Copy v3 /data and /Reports folders to v4 install - - Rename the /data folder to /datav3 - - Run MigrateData3to4 - - Done! -- Removed previously deprecated web tags - CO2-24h, CO2-pm2p5, CO2-pm2p5-24h, CO2-pm10, CO2-temp, CO2-hum -- Loading dayfile now continues on error and reports total errors - only the first 20 errors are logged -- You now only set the Ecowitt MAC/IMEI address in one place for the various station types - - In Local API settings for GW1000 type - - In Cloud Access API for Cloud and HTTP station types - - -Fixed -- Problems when using a 9am rollover in the records editors for values from the monthly log files -- Select-a-Period charts not respecting the interval dates: Air Quality, CO₂, Soil Moisture, Leaf Wetness -- Calibration Limits not changing when the user changes units - eg initial install -- Potential fix for corruption at the end of all data log files when shutting down -- Error that the username is not set when sending email to a server that requires no authentication -- Improvement to GW1000 API reconnects -- Improved web socket initial connection to send data immediately on dashboard/now/gauges connection -- Fix for soil moisture conversion from percentage to cb in Weather Cloud uploads -- Reload dayfile can now only be run as a single instance -- Improvements to Davis WLL wind handling when: - - Transitioning from catch-up to live running - - No broadcasts are received -- Davis WLL improved recovery from loss of broadcast messages -- Spike/limit improvements - - - -############## Cumulus MX version 3.0 below here ############## - -3.28.6 - b3283 -—————————————— -- Fix davis Cloud station decode of VP2 sensor current data (again) -- Fix crash in Growing Degree Days & Temperature Sum graphs for a particular circumstances - - -3.28.5 - b3282 -—————————————— -Fixed -- Bug in rainfall jump detection, was triggering at 1 mm! - - -3.28.4 - b3281 -—————————————— -New -- The monthly log file check for corruption is extended to the Extra and AirLink logs if they are in use - -Fixed -- Error that the username is not set when sending email to a server that requires no authentication -- Some Davis VP2 stations current data not working with the WeatherLink Cloud station type -- Fix for soil moisture conversion from percentage to cb in WeatherCloud uploads -- Fix EcowittCloud not applying historic piezo rainfall - - - -3.28.3 - b3280 -—————————————— -New -- Adds Ecowitt local GW1000 API to the web tag <#StationFreeMemory>. Known to work with GW1100 (v2.3.1) and GW2000 (v3.1.1) so far -- Adds a new web tag <#ExtraStationFreeMemory> to avoid clashes with the main station and a Ecowitt HTTP station used for extra sensors -- Adds a check for the raincounter suddenly increasing (more than 25mm or 1 inch) - -Fixed -- Ecowitt Cloud decoding of Soil Moisture values -- Ecowitt Cloud add missing decode of CO2 temp/humidity values -- Crash for new installs when the monthly log file does not exist -- Setting Ecowitt custom server settings when used as an Extra Sensor station -- The Monthly Records editor values from the log file showed the previous months values for monthly totals like Total Rainfall etc. - - -3.28.2 - b3279 -—————————————— -New -- Adds two new web tags <#StationFreeMemory> and <#StationRuntime> - StationFreeMemory: Shows the station free memory for Davis WLC, Ecowitt GW1100/20000 (using HTTP) - StationRuntime: Shows the station uptime in seconds for Davis WLL/WLC, Ecowitt GW1100/20000 (using HTTP) -- Now implements a basic check of the latest monthly log file on start-up to remove any nulls from the end - -Changed -- Custom Minutes MySQL commands can now have the interval defined per command rather than globally - -Fixed -- Ecowitt Cloud station: Fix some errors decoding Ecowitt cloud API. -- Ecowitt Cloud, no longer requires a Pro subscription to pull current data. Quite an extensive rewrite! -- Extra Files using log filename templates missed the last log entry at month rollover as they switched to the new month immediately -- Fix missing units in alarm email messages -- Custom Timed MySQL commands sometimes fired immediately after saving the configuration -- Ecowitt historic catch-up was missing CO2, CO2 pm, CO2 T/H, leaf wetness, and extra T/H dewpoint values - - -3.28.1 - b3278 -—————————————— -New -- PHP upload max connections per server now configurable, the default value is reduced from 20 to 5 -- Extra Web Files upload of log files now supports incremental uploads (works for all upload types: PHP, SFTP, FTP/FTPS) - - New tick box column on the settings page "Incremental" - - Can be used with any text file that "grows" a line at a time -- Upload Now! page now has a tick box to force upload of incremental log files fully - -Changed -- Web tag <#OsVersion> now returns the OS description. This is better on Windows and appears to be the same on Linux - -Fixed -- Final(?) fix for remove space from dates -- Add a check for null MQTT topics -- A bug in the MySQL timed + repeat settings -- Error save Program Settings > RemoveSpaceFromDateSeparator -- Fix wrong DST status issue that seems to affect some Synology NAS boxes -- Fix record editors when editing records that have a time format of "Mon Year" -- When Davis WLL detects a missed high wind gust it is now also added into recent wind data to be included in the calculations - -Package Updates -- FluentFTP (revert back to official release) -- MQTTnet -- MySqlConnector - - - -3.28.0 - b3269 -—————————————— -New -- Adds the option to clear the latest error log -- Adds User Defined Alarms - - Trigger an alarm when any numeric or boolean web tag value exceeds or falls below a set value - - Boolean web tags evaluate to 0 or 1 (false, true), so are treated as numeric -- You can now change the built-in alarm descriptions in the Locale Strings settings -- Adds a new web tag <#EcowittVideoUrl> to get the URL of the latest Ecowitt camera timelapse video - -Changed -- Warning message from WLL station about spurious broadcast messages changed to a Debug message -- All time low wind chill web tags were miss named wchillH and TwchillH. They are now wchillL and TwchillL - - The default web site page record.htm has been changed to reflect this - - As has the websitedataT.json file -- Calibration and Limit values are now displayed and set in your native units -- NOAA reports now produce an error rather than skipping over if a duplicate entry is detected in the dayfile -- Reload dayfile now produces an error rather than skipping over if a duplicate entry is detected in the dayfile -- The default path in the sample cumulusmx.service file has been changed to /opt/CumulusMX -- Upload.php amended to change the maximum permitted clock difference between MX and the remote server to 20 seconds (was 10) -- The MQTT config files are now only read at start-up or when the MQTT Settings are saved -- The MQTT settings have been removed from Internet Settings to their own page -*** BREAKING CHANGE *** -- The MQTT interval data sending has changed. Previously all topics sent at the same fixed interval. Now you can specify an interval per topic in the MQTT config file - If you are using MQTT interval topics, then please the ReadMe.txt file in the /mqtt folder for details of the changes you need to make - -Fixed -- Add missing New Record and FTP alarm info to the dashboard data stream -- Some AirLink wl.com API calls not using the correct API details when configured as Standalone -- Indoor Temperature and Humidity spike removal not working -- Program Setting 'Remove space character from date separators (if present)' now applies to all output -- The Ecowitt API MAC address field was not allowed to be blank in the Wizard or station config -- Wind Gust values not being recorded as high values with moderate spike values set and wind speed in mph or knots -- Fix average wind speed values from the day file in the records editors - -Package Updates -- MailKit/MimeKit -- MySqlConnector -- ServiceStack.Text -- SQLite -- MQTTnet -- FluentFTP (customised CMX version until new a build is released) - - -3.27.1 - b3263 -—————————————— -New -- Two new web tags for the Latest Error. These tags will encode the string for either HTML or JavaScript: <#LatestErrorEnc>, <#LatestErrorJsEnc> -- New web tag for monthly wind run total <#windrunmonth> - - Without parameters it returns the total for the current month so far (excluding today) - - You can add parameters "year" and "month" to return the value for a particular month. Eg. <#windrunmonth year=2022 month=3> - - Like the existing <#windrun> it also accepts the "unit" parameter to return the value in miles, kilometers, or nautical miles -- PHP upload advanced option to disable the use of GET for small files and fallback to using POST for all files -- Ecowitt stations can now map the indoor temp/hum values to the primary outdoor sensor (primarily of use for 4G stations that do not allow extra T/H sensors to be added) -- New web tags for <#RecentApparent>, <#RecentIndoorTemp>, <#RecentIndoorHumidity> - -Changed -- All third-party upload issues are now classified as Warnings -- Third party components for FTP(S) and SFTP updated -- If Cumulus.ini needs rewriting it is now overwritten, if it needs recreating it is first truncated to zero length. Previously the file was deleted and recreated in both instances which - made linking the file for example in Docker images impossible. -- The records editors now set both value and date/time when you click on the record from the dayfile for monthly logs -- The records editors now accept y/n key presses to update records or cancel - -Fixed -- "Object is not initialized" error from the API when the station is not yet ready -- MX crash on start-up if the SQLite database has certain corruptions -- Fix for PHP uploads when no compression is supported by the server (or the network was down when MX started) -- VP2 error message "No Ack in response to DMPAFT" on start-up -- Ecowitt Cloud API settings are no longer mandatory for Ecowitt station (bug introduced in 3.27.0) -- Fix for error 500 in upload.php when debug is enabled (only occurred on some servers) -- PHP uploads using HTTP GET were uploading to the same destination file on one server. Added an Internet advanced option to disable the use of GET and fallback to using POST for all files -- Ecowitt Extra Cloud sensor station type not starting as expected -- Ecowitt Extra Cloud sensor station can now pick up the latest web cam URL -- Some new record alarms not setting the email message correctly -- Fix for wind spike testing applying unit conversion multiple times in some circumstances -- Davis VP2, if requested changing the logger interval is now done after reading catch-up data so the archive is not cleared before catch-up -- Fix error 500 in the config Wizard for Davis Cloud stations -- Fix Wizard and Settings screens did not allow the entry of an IMEI number for the Ecowitt Cloud API device ID -- Fix some potential errors in the dewpoint calculations on Ecowitt stations when using a mapped primary temp/hum sensor -- Fix for zero length string error when parsing web tokens -- Perform a second chance PHP upload compression test at upload intervals if no compression detected at start-up - - -3.27.0 - b3257 -—————————————— -New -- Adds a new station type of Ecowitt Cloud. This is intended to be used with those Ecowitt console that no not have a local API, cannot use the HTTP protocol, or are located remotely. - - It grabs the data from the Ecowitt cloud servers. This data is updated once a minute and is slightly less rich than the locally available API data. - - This station can also be used as a source of Extra Sensor data for your primary station - - It also makes available the URL of the latest camera on the Ecowitt cloud server via the new web tag <#EcowittCameraUrl> - - You can use the tag "" in the URL field of HTTP Files to download the latest image -- Adds a new station type of Davis Cloud - - It grabs the data from the Davis WeatherLink cloud servers. The data is updated according to your subscription model - - Pro+ = 1 minute updates - - Pro = 5 minute updates - - Free = 15 minute updates -- Adds new Select-a-Period graphs to the dashboard interface. - - Similar to the existing Select-a-Graph which only show recent 1 minute data, the new charts show data at your logging resolution, but you can select any start and end dates from your historic data. - - Note: selecting a large date range can be quite slow as the data has to be read from the log files -- You can now regenerate ALL missing NOAA reports from either the Month or Year NOAA screens - - Note: Unlike the (Re)Generate button, the new option will not overwrite existing report files, it will only create missing ones -- There is a new Latest Errors display in the dashboard - Utils | Latest Errors - - By default it displays the last 50 Warnings and Errors encountered by CMX, if you are seeing too many warnings that are expected, you can opt to log only errors in Settings | Program Settings | Logging Options -- The Cumulus settings can now be 'secured' with a username/password. NOTE: Because Cumulus does not use HTTPS these credentials can be snooped on the network, the security provided is limited. - - This feature has required some minor changes to the Cumulus API: wsport.json, version.json, dateformat.txt, and csvseparator.txt have moved to a new path /api/info/ -- Add forwarders to Ecowitt Extra Sensors station type -- Add an integrity check to the SQLite database on start-up - - -Changed -- Internal changes to update the Visual Studio project files from the previous legacy version -- Additional keep-alive headers added to PHP uploads -- PHP upload now uses HTTP GET for data sizes < 7000 bytes - - IMPORTANT: You must update the copy of upload.php on your web site with the latest version in this release -- Dashboard gauges now follow the general dark theme -- Adds UV-I to dashboard today/yesterday -- PHP Upload now closes the connection before it errors - if the host server sends a maximum number of allowed requests per session -- The following web tags have been deprecated and will be removed in some future release: - CO2-24h - CO2-pm2p5 - CO2-pm2p5-24h - CO2-pm10 - CO2-temp - CO2-hum - Please switch to the new names: - CO2_24h - CO2_pm2p5 - CO2_pm2p5_24h - CO2_pm10 - CO2_temp - CO2_hum - - -Fixed -- Fix for Davis average wind spike on restarts with catch-up -- Bug in Fine Offset console logger interval setting -- Error saving Locale Strings page - new all-time record alarm was missing -- Default Web Site: - - Fix Select-a-Charts for Davis soil moisture sensors limiting the graph to a maximum value of 100 cb - - webfiles\js\selectachart.js - - Fixed some rounding errors when converting windspeeds to non-native units - - webfiles\lib\steelseries\scripts\gauges.js -- Ecowitt HTTP Station - fixed piezo rain rate not being decoded -- All stations, fixed rain rate being recorded as zero in the recent graphs when MX was calculating the rate rather than the station supplying it -- Wind average calibration is once again applied to live data when the option for MX to calculate the average is enabled -- Davis WLL 10 minute Gust speeds when no broadcasts are being received -- Error 500 on Locale Strings settings page -- Fix CO₂ sensor being visible on graphs by default -- Many Alarms were reverting the Latch Hours value to an integer on restart -- If the graph data display options are changed whilst PHP incremental uploads are in use, then the uploads may start failing -- Web tag #rmidnight was being reset before the EOD processing (midnight rollover) -- Fix bug setting HTTP Ecowitt station custom server -- AirLink Health error messages on start-up when Airlink is associated with the main station -- Davis VP2 applying average wind speed calibration to the gust value in LOOP2 packets - - -3.26.0 - b3248 -—————————————— -New -- MQTT has a new behaviour - Only update a topic if the data has changed - - If you add a new property called "doNotTriggerOnTags" to a topic, you can specify a list of web tokens. Adding this property with at least one entry triggers the new behaviour. - The web tokens in the list are excluded from the comparison process, so for instance if you include a timestamp in your topic, you can exclude this from the change detection. - For example: - { - "topic": "CumulusMX/TempUpdate", - "data": "\"time\":\"<#timehhmmss>\",\"temp\":<#temp rc=y>", - "retain": false, - "doNotTriggerOnTags": "timehhmmss" - } - Which now will only send a new MQTT message when CMX first starts, then only when the "#temp" value changes. -- Adds new alarm for any all-time record being set -- Adds new alarm "Web upload errors" for errors in the FTP/PHP upload processes -- Adds the ability to append free text comments to the wxnow.txt file. You can also include web tags -- New web tag for recent rainfall - <#RecentRain> - it takes the same parameters as the other RecentXxxx web tags, and defaults to today's total if the parameters are incorrect or no recent data is available -- New web tag for Ecowitt sensor radio reception strength - <#EcowittReception> - - By default it returns a string of comma separated sensor names and value. eg "WH80=4,WH41CH1=3" - - You can add a parameter "format=json" - <#EcowittReception format=json> - and it will return a valid JSON string. eg. {"WH80":4,"WH41CH1":3} - - The values are updated every 20 minutes along with the battery info -- The existing web tag <#txbattery> now also supports the format=json parameter which functions the same as #EcowittReception tag above -- New alternative name for the existing web tag <#GW1000FirmwareVersion>, you can now use <#EcowittFirmwareVersion> -- Adds Custom MySQL commands for when Cumulus starts up. These will be executed after the station is initialised, but before it starts any historic catch-up or live processing - -Changed -- Various third party libraries updated to current versions - - MailKit, MQTTnet, MySqlConnector, ServiceStack -- PHP upload now sends a text/plain content header -- Davis WLL and Airlink now use the new simplified access method to weatherlink.com V2 API -- MX now persists the internal recent wind data across restarts so gust and average speeds are consistent on a quick stop/start of MX - -Fixed -- PWS passwords are now URL encoded -- Solar irradiation and theoretical solar are now stored as integers everywhere -- Extra end of day files not being processed on copy -- Davis VP2 stations creating incorrect values for Wind Run and Chill Hours if some logger data is missing -- Instromet station now forces MX to calculate the average wind speed -- FTP logging errors when FTP logging is already enabled at start-up -- Updating MySQL on DayFile edits was erroring with a blank SQL statement -- Davis stations spike in average wind speed on stop/start of CMX - - - - -3.25.2 - b3245 -—————————————— -Fixed -- Davis WLL/AirLink not doing historic catch-up or downloading stats from weatherlink.com - - - -3.25.1 - b3244 -—————————————— -New -- Adds the station name to the main dashboard page and email subject - -Changed -- NOAA report selectors no longer limited to years greater than 2000 - -Fixed -- Tempest station now forces MX to calculate the average wind speed -- Leaf wetness sensor name changes not being saved in Locale strings -- Leaf wetness graph data empty for Davis sensors -- Average wind speed calculation for Davis VP, Davis WLL (when calibration applied), Imet, Tempest, WS2300 stations -- Another average wind speed calculation for Davis VP and Davis WLL (when broadcasts are not being received) -- Dashboard Local Time block - again! -- Zero length string error processing web tags -- Graph colours for PM2.5 and PM10 getting set to the same value -- Uncaught exception (TaskList has null entry) in PHP uploads -- 100% CPU bug in the new HTTP files on first processing of each file -- Revert TLS1.2/1.3 changes in v3.25.0 -- Extensive HTTPClient code optimisations -- Fix year bug reading from monthly log in NOAA monthly report generation -- Fix spurious TokenParser messages from various external task executions - - -3.25.0 - b3241 -—————————————— -New -- Add data forwarding for HTTP (Ecowitt) stations (EXPERIMENTAL for now) -- Adds Leaf Wetness as a recent and select-a graph option - also added to chart colour picker -- Local API, graph data requests now support the URL parameter start=nnnnn where nnnnn = Unix timestamp for first data values -- The number of concurrent PHP uploads can be configured in Internet Settings | Web/Upload site | Advanced Settings (defaults Windows=4, Mono=1) -- Non-linear calibrations can now be applied to all applicable values -- A new download file/upload to web option:- HTTP Files - - This allows you to download a file from any http URL and either save it locally, or upload it to your web site - - This can be done at any interval, starting when CMX starts - - Or you can specify a start time and interval. For example you want to download a forecast image that is updated at 06:00, 12:00, 18:00 - - You could specify an interval of 6 hours starting at 06:01. The image will then be downloaded at 06:01, 12:01 and 18:01 -- Custom MySQL commands now have a new option of Timed updates. These work the same as the HTTP Files timed uploads - -Changed -- Deprecate the StartDate entry in cumulus.ini. The new entry is called StartDateIso and has a consistent format across locales of YYYY-MM-DD - The existing entry will be migrated and removed automatically -- Switched to the latest current build of FluentFTP as a fix is now been incorporated for Mono errors - - This has meant a change to the FTP logging processes. FTP logs are now created in the MXdiags folder, and cycled like the main CMX log files -- Adds a retry on failure for PHP Uploads -- PHP Uploads now restricted to prevent excessive process generation under Mono -- All HTTPS operations (Third Party, Custom, Catch-up from cloud etc) now use TLS 1.2 and TLS 1.3 only - -Fixed -- Real-time FTP/SFTP upload of realtimegauges.txt could corrupt concurrent FTP/SFTP interval uploads of processed files -- Extra web files was performing the EOD copy before the daily graphs had been created -- Possible fix for Child process count increasing when using PHP upload -- The monthly and extra data log file editors now show the correct date for 9/10am rollover configurations before the daily rollover -- Real time SFTP not reconnecting when SFTP object is null or not connected -- The following processes now abort processing if they are called before the core met data is available - Custom Interval Logs, Custom MySQL: Realtime, Interval, and Timed commands -- Real time FTP login was sometimes being attempted even if FTP was disabled completely - occurred if the real time FTP option was left enabled -- 24 hour times on the dashboard showing as 24:nn after midnight rather than 00:nn - - - -3.24.2 - b3235 -—————————————— -New -- Extra Files now allows you to use the variables thru in the source and destination fields to specify custom log files to be uploaded/copied -- Copy/Upload Now! gets a new option to transfer the latest NOAA monthly/year reports - -Changed -- Copy/Upload Now! the upload of full rather than incremental versions of graph data files is now optional - -Fixed -- Davis VP2 logger suppression of full logger downloads if CMX is stopped and immediately restarted fractionally after a logging interval -- Extra Files with EOD flag having "tmp" appended to all source files -- Not running as a 64 bit application on Windows -- NOAA report PHP uploads were no using the UTF8 flag -- Davis WLL using "stale" current wind gust data -- Daily graphs not being uploaded/copied after day reset - - -3.24.1 - b3234 -—————————————— -New -- Implements the dashboard time format override - just on the Dashboard and Today/Yesterday screens for now - - Configure in Program settings | Culture Over-rides -- Adds Station Pressure support for Ecowitt API, Ecowitt HTTP, Ambient HTTP, and EasyWeather stations - -Changed -- Extra sensor graph data files now use the customised sensor names -- The "recent" daily data graphs (daiyrain.json, dailytemp.json, sunhours.json) are now treated like End-of-day files and only created at start-up and daily rollover -- The graph config files availabledata.json and graphconfig.json are now only uploaded at start-up or on config a change -- Updates ExportMySQL to v1.8.0 to be compatible with v3.24.0 or later - -Fixed -- Issues in 3.24.0 with Select-a-graph not working on the dashboard and default web site -- Extra sensor names not saving in Locale strings -- Extra Dew Point graph data file was empty -- The initial PHP Upload of graph data files sent 7 days of data regardless of the Graph Hours setting - - - -3.24.0 - b3231 -—————————————— -New -- Strings.ini is now created and used by all instances, it has also added to the backup routine -- Adds customisable graph data series colours to the dashboard and default web site -- Two new web tags for signed string values of pressure and temperature trends - <#presstrendsigned> - <#temptrendsigned> -- A new web tag for the station pressure (the absolute pressure reading) <#stationpressure> -- The data visibility has been changed from a simple off/on to a tri-state. You can now select between Off|Visible|Dashboard Only - - Visible means the data will be shown on both the local dash and the default web site - - Dashboard Only means the data can only be viewed on the CMX local interface. - You may wish to use this to hide sensitive data like indoor temperatures on the public web site which can indicate if you are home or not - - Any data previous flagged as Visible, will migrate to the new Visible state -- Extra Sensor graph data can now be sent to the default web site -- You can now send an additional four extra temperature readings to Weather Underground -- New PHP upload alternative to FTP/SFTP - - This uses a secure PHP script on your web server to upload the data rather than using an FTP service - - The graph recent data files are sent incrementally using this mechanism, cutting bandwidth usage - - On starting CMX a full copy of each data file is sent - - Thereafter only the new data for the interval is sent, and the PHP upload script appends this to the existing file, and trims out old data -- The generation of local copies of the data files is now optional - - You can deselect the "Generate" option if you do not require a file to be written to the CMX folders. This will reduce disk I/O, a concern for rPi users with SD cards -- Adds the HighCharts accessibility module to all charts with the default basic settings. See:- https://www.highcharts.com/accessibility/ -- Adds the simplified NOAA cooling/heating day calculation to the NOAA reports settings -- Cumulus MX will now run as a 64 bit application on 64 bit operating systems -- You can now use the Ecowitt WS90 haptic rain sensor for "Is Raining" if you are using a tipping bucket sensor as the primary rain sensor - - See Station Options > Common Options - -Changed -- Some entry errors in the settings screen used to output a invalid pattern and regex message. They now output an English language message -- The Dashboard page can now shows times in your locale format (12/24 hour) - See Program Settings|General Options -- Moved the existing Graph and Display settings from Station Settings to a new Display Settings page -- Custom Logs settings screen now shows examples with the correct CSV separator for the CMX locale -- Tempest station now forces the UDP listening port to be opened in shared mode -- The "Stop second instance" option has been rewritten. It now only prevents two instances *using the same port* from running - - This change means for it to work ALL running instances must be v3.24.0 or later -- Alarm email sends are now retried twice on failure - -Fixed -- Adds missing Wind Chill from the historic graphs on the dashboard and default web site -- Now ignores AirLink zero values when the percent in last hour is also zero. This occurs during a restart of AirLink -- Third party custom rollover URLs overwriting custom minute values on save -- The monthly data editors were not handing start/end dates of the first of the month correctly when using a 9am rollover -- The <#altimeterpressure> web tag was defaulting to using 1 decimal place for all units -- Dashboard menu wrapping -- Davis VP2/Vue LOOP2 absolute pressure values < 20 inHg are now ignored. Altimeter pressure is set to sea level pressure, station pressure = 0 -- Extra temp/hum/dewpoint not being saved for sensor #10 -- Ecowitt Local API failing to reconnect after network outage - -Removed -- Local API calls: - /api/extra/leaf.json - /api/extra/leaf4.json -- Redundant web tags <#leaftemp[1-4]> - - -3.23.1 - b3221 -—————————————— -New -- Adds dew point to extra sensor graphing -- Add extra sensors to the dashboard select-a-chart - -Changed -- Davis WLL now fully falls back to the local API current conditions for rain and wind data if broadcasts stop being received for 30 seconds - - There is now a new Advanced setting for the WLL: Trigger DataStopped on Broadcast stop - Previously if broadcasts stopped being received, a full DataStopped condition and alarm was triggered - Now CMX falls back to using the local API for all data - This new configuration setting allows you to set the DataStopped only when both broadcasts AND local API responses stop being received - -Fixed -- Extra sensor graphs stopping at month end -- Start-up/rollover backups were copying Custom logs to the CMX root folder rather than the backup folder -- Davis WLL not updating the gust value if broadcast data stops being received -- AWEKAS: - - Missing "&" on soil temperatures - - Producing an exception message when it has already logged the response as invalid - - -3.23.0 - b3220 -—————————————— -New -- You can now use web tags in all external program parameter strings (except start-up) - - Realtime, interval, daily, shutdown, and alarms -- Indoor temperature and humidity now have spike checking, and linear calibration -- The Data Logs viewers and Purge MySQL Cache now have multi-select enabled for deleting entries -- Two new web tags for the last MySQL insert times that can be used in custom MySQL commands - <#MySqlRealtimeTime> = time of last real-time table insert - <#MySqlIntervalTime> = time of last interval data table insert -- Most weather value web tags now perform unit conversion. Specify the web tag parameter: unit=[xx] - Where [xx] is one of the following (case insensitive): - Temperature: C | F - Pressure : hPa | mb | kPa | inHg - Wind : mph | kph | ms | kt - Rainfall : mm | in - Wind run : km | mi | Nm - You may also need to override the number of decimal places when converting. - For example, when converting native mb or hPa to kPa or inHg, you would also need to add "dp=2" - <#press unit=kPa dp=2> -- Adds graphing of some extra sensors on the local dashboard interface - Implemented for: Extra Temperatures, Extra Humidity's, Soil Temps, Soil Moistures, User Temps, and CO₂ - -Changed -- Lots of third-party modules updated - - EmbedIO, MailKit, MimeKit, MQTTnet, MySqlConnector, ServiceStack.Text -- Davis WLL: Badly formatted broadcast messages are now dumped to the log file, and printer utility broadcasts logged separately -- Removed fictional Leaf Temperature values from the Extra Sensors screen -- The Rain Today editor now immediately updates the rainfall this year/month values -- Now trims all text input fields in the settings pages - -Fixed -- Spurious alarm cleared messages when the alarms are not enabled -- Davis VP2 logger downloading entire contents if CMX was restarted before the next console log interval -- Stop second instance on Linux, this now uses a lock file to prevent a second instance from running -- Fix Custom HTTP intervals from trying to process null entries -- Davis WLL - - Missed broadcast package counter not being incremented - - No longer processes broadcast messages twice or more on systems with multiple network interfaces - - Improved collision avoidance between broadcasts and API requests - - Possible fix for the memory leak when running under Mono -- Ecowitt - - Fix for missing average temperature calculation during Ecowitt historic catch-up - - Historic catch-up is now cancellable with Ctrl-C - - -3.22.4 - b3215 -—————————————— -Fixed -- Removing last entry from Third Party custom HTTP lists -- Disallow negative solar rad values -- Ecowitt piezo rain sensor (WS90) now supports storm event -- Davis Vantage VP2/Vue stations no longer download the entire logger contents on start-up when the station logging interval does not match the CMX interval -- Catch errors in La Crosse WS2300 data handling - specifically fixes reported crashes in forecast data -- Added a moon image cache-buster to the default website index page - just upload the new index.htm file - -New -- Adds start-up and shutdown tasks - see Program Settings - - -3.22.3 - b3214 -—————————————— -Fixed -- Updates and fixes to the MQTT library -- Crash in the WLL station fetching Available Sensors from wl.com when it is unreachable -- Recent data graph JSON files may contain differing numbers of records - -Changed -- The records editors now cope with blank fields (above field 15 - Current Gust) in the monthly log files - - -3.22.2 - b3213 -—————————————— -Fixed -- Broken 24hr rain and rain rate in v3.22.1 - - -3.22.1 - b3212 -—————————————— -Fixed -- NOAA Reports pages not working in all locales -- RainRate sometimes giving wild values on start-up when CMX calculates rate -- Amusing typo in default web site today/yesterday page - - -3.22.0 - b3211 -—————————————— -New -- Adds the ability to create custom data log files, both interval and daily -- Ecowitt local API station now dumps the station clock drift to the log file at start-up and every day at noon - -Changed -- MailKit and MimeKit updated to latest versions -- The web tag <#TimeJavaScript> now provides a value truncated to the current second - - -3.21.2 - b3206 -—————————————— -Fixed -- Pressure/Temperature trend alarms triggering before the first data has been received - - -3.21.1 - b3205 -—————————————— -Fixed -- Custom MySQL settings - - -3.21.0 - b3204 -—————————————— -NOTE: When running under Mono, this release REQUIRES Mono version 6.12 or later for email functions - -Fixed -- FTP delete before upload aborting if the initial delete fails due to file not existing -- Local file copy giving "index out of bounds" error - -New -- Failed MySQL commands are now can now be individually edited/deleted - -Changed -- Failed MySQL commands are now stored in the SQLite database to persist across Cumulus runs -- Third party components updated - - - -3.20.1 - b3203 -—————————————— -Fixed -- Fix crash in Alarms with some station types -- Change the MySQL table updates to compare column names rather than simple counts -- Fix the All Time, Monthly, This Year records editors not allowing 24 hour rain values to be changed -- Fix Dayfile editor not updating ChillHours and 24 hour rainfall values in the MySQL database -- Fix Dayfile editor inserting 9999/-9999 values into the MySQL database when values are missing - -New -- Adds a new option to the email server settings to ignore certificate errors - -Changed -- The station latitude and longitude are now stored internally as decimal values. This means there will be no loss of precision when storing locations entered as decimal values via the - configuration wizard, or directly in the Cumulus.ini file - - -3.20.0 - b3202 -—————————————— -Fixed -- Buffering of failed MySQL queries -- MySQL library update to fix crashes with a null reference -- NOAA Monthly report selector showing two "March"s and no February on the first day of each new month -- Alarms being triggered during data catch-up. These are now suppressed - -New -- Add custom actions for alarms. Call a script or external exe etc. -- Add a This Period records display like C1 -- New option to exit Cumulus MX after a data stopped condition - Program Options > Shutdown -- All DateTime web tags now accept format strings of "Unix" and "JS", if either of these is supplied the datetime will be output as either a Unix or JavaScript timestamp -- Rain 24 hour: - - Added to day file, records 54, 55 (value, time) - - Updated dayfile header. Also abbreviated and changed to CSV from PipeSV - - Added to all the record editors - - Added to MySQL. See new feature below for updating your existing Dayfile table. There is also a MySQL script in the MXutils folder to alter existing table - - New web tags: - <#rain24hourTH>, <#Train24hourTH> - <#rain24hourYH>, <#Train24hourYH> - - Default web site updated to show all the relevant records -- Adds the ability to search in the Dayfile viewer/editor -- FTP and FTPS now has an option to enable automatic detection of the connection settings - - This and another new setting Ignore Certificate Errors are in Internet Settings | Web/FTP Site | Advanced Settings - - NOTE: If you use an FTP server without a public certificate, you *MUST* enable the setting. This is a breaking change -- Custom HTTP calls (seconds, minutes, rollover) can now each have up to 10 URLs -- Custom MySQL Uploads (seconds, minutes, rollover) can now each have up to 10 commands -- The MySQL Settings page gets new functions for updating existing tables by adding columns to match the current schema -- A new "Utils" menu on the Dashboard interface - - The FTP Now function has been moved there - - New option to reload the Dayfile into CMX from file. CMX holds a copy of the Dayfile in memory, if you edit it externally, use this to refresh the values in CMX without restating - - New option to purge the failed MySQL command queue. If you have failed commands (normally custom commands) that will never work, then you can remove them from the failed retry queue without restarting CMX - -Changed -- Alarm latch time hours are now decimal values rather than integer -- Moon Rise/Set web tags now have the date set to the current day -- The records editors now work in your locale date/time formats, they also now report any errors setting values -- The Monthly records editor month tabs have improved accessibility -- The default web site monthly records page month selection button have improved accessibility -- Local API, removed BOM from all API responses -- The date pickers for the log file view/editor pages and the new This Period now display the date in the CMX locale format -- The Cumulus.ini file now only contains FTP ExtraFiles entries that have either a local or remote filename entered (potentially removes 800 lines from the file!) - - -3.19.3 - b3196 -—————————————— -Fixed -- Error when sending an Is Raining alarm email - - -3.19.2 - b3195 -—————————————— -Fixed -- Suppress trying to sync solar data on FO stations with solar during night time hours -- Fix problem processing buffered failed MySQL commands. More query string issues excluded from buffering -- Default gauges.htm page CSS for the Wind Rose fixed - -New -- Adds the ability to use leaf wetness sensors to trigger the "Is Raining" value and optional alarm. Configure this via Station Settings > Common Options -- Adds a hash file for the distribution files - hashmd5_.txt - - -3.19.1 - b3194 -—————————————— -Fixed -- Dashboard occasionally showing zero values when using Ecowitt/Ambient extra sensors with the extra HTTP station feature -- Sunshine Hours graph data had badly formed JSON -- Fine Offset station getting into a data read synchronisation loop when synchronisation fails. It will now give up after two attempts -- Potential fix for FTP client locking files on connection error - -New -- Added new midnight temperature range tags for today and yesterday - <#tempMidnightRangeT> <#tempMidnightRangeY> -- Adds new "Is Raining" alarm, triggered either via a Hydreon RG-11 device, or rain rate > 0 - The alarm has a new associated web tag <#IsRainingAlarm> -- Adds a new option under Station > Common Options > Advanced Options to use station rainfall (rate > 0 or tip occurred) to trigger IsRaining - - -3.19.0 - b3191 -—————————————— -Fixed -- Ecowitt Extra Sensors setup error 500 -- Broken NOAA reports in dashboard -- Occasional web socket disconnection in the Dashboard -- Default web site pages not always fully loading - updated /js/setpagedata.js -- Ecowitt Local API station failing to start-up correctly when auto-discovery is enabled and the station fails initial discovery -- Fine Offset console read synchronisation is fixed/changed - -New -- Added the Today/Yesterday page to the default web site - -Changed -- Migrate the log file viewers to the new date picker -- Default web site menu system - add accessible sub-menus (setpagedata.js) -- Date selection in NOAA reports -- Local API: NOAA reports are now returned as plain text rather than a JSON array of strings -- Dashboard now populated with data on initial page load (also Now and Gauges pages) -- Davis WLL: Querying the WLL current conditions now avoids times a broadcast is due to be sent as the WLL cannot do both at once - -Removed -- Twitter be gone! Cumulus MX no longer supports Twitter directly - - - -3.18.0 - b3190 -—————————————— -Fixed -- Adds Ecowitt Wh40 battery state decode -- Increase the sensitivity of the rain counter reset and make it "unit aware" - -Changed -- Lots of third party libraries updated to the latest versions. This should improve SFTP and FTPS connectivity amongst other things -- Weather Diary editor changed to improve accessibility of the Date Picker -- Records editors changed to improve accessibility, they now also use pop-up editors -- Records editors now have the ability to click on the values/timestamps of the log files and have them copied to the record value/timestamp - -New -- Additional web tags for today and yesterday high and low temperatures using a midnight rollover. Obviously if you use a midnight rollover - for your met day then these tags will return the same values as the existing hi/lo temp web tags. - <#tempMidnightTH>, <#TtempMidnightTH>, <#tempMidnightTL>, <#TtempMidnightTL> - <#tempMidnightYH>, <#TtempMidnightYH>, <#tempMidnightYL>, <#TtempMidnightYL> - - -3.17.0 - b3184 -—————————————— -- Fix: Cloud base being set to large value at start-up -- Fix: Reading lightning distance from today.ini -- Fix: Potential crash obtaining the local IP address - -- New: Adds a PWS Simulator station type for testing or trial purposes - -- Change: The "live" dashboard screens now refresh whenever new data is received, or every five seconds -- Change: The "Speed for average calc" station option has been moved from Common Options to Common Options | Advanced Options -- Change: The Ecowitt stations now force the "Speed for average calc" option to be enabled at start-up -- Change: Slightly improved solar position calculations - - -3.16.1 - b3183 -—————————————— -- Fix: Error message about Ecowitt sensor mapping when saving the station settings for non-Ecowitt stations -- Fix: Some Ambient stations are not sending the yearly rainfall total - they send total rain instead -- Fix: Ecowitt stations with a Blake-Larsen were erroneously adding CMX calculated sunshine during catch-up -- Fix: Potential issue whereby Ecowitt historic data download could get stuck in a loop downloading the same block - -- Change: Ecowitt Local API stations with WS90 sensors now get the update rate set to 8 seconds (previously 4) - - - -3.16.0 - b3182 -—————————————— -- Fix: Fix mislabelled July solar transmission factors to June -- Fix: Web tags <#chillhoursToday>, <#chillhoursYest> when chill hours not increasing -- Fix: Alarm settings could not be saved unless a valid from-email address was entered. Now it is only mandatory if an alert has the email option checked -- Fix: Ecowitt catch-up not processing rain data - and nobody has noticed! -- Fix: CPU temp check on Linux? -- Fix: Start-up PING is now run in a separate thread so if it hangs CMX can continue -- Fix: Ecowitt Local API station not performing a battery check every 20 minutes -- Fix: Longest dry/wet web tags from outputting -9999 values, if uninitialised they now output "--" - -- New: Ecowitt WN34 sensors can now be mapped from User temp to Soil Temp -- New: Ecowitt rain sensor is now selectable between tipping bucket and piezo sensors - both Local API and HTTP Ecowitt protocols -- New: Adds ability for CMX to configure Ecowitt custom server when using it for Extra Sensors -- New: HTTP Ecowitt stations, Cumulus MX will now configure the custom server config for you - optional -- New: Ecowitt stations (Local API and HTTP), adds the ability to override the default outdoor temp/humidity values by specifying an extra T/H sensor channel -- New: Adds last 24 hours rain to the dashboard "Now" page -- New: Adds records for 24 hour rainfall - This Month, This Year, Monthly, and All Time - - New web tags: - <#HighRain24HourRecordSet> - <#ByMonthRain24HourH>, <#ByMonthRain24HourHT> - <#MonthRain24HourH>, <#MonthRain24HourHT>, <#MonthRain24HourHD> - <#YearRain24HourH>, <#YearRain24HourHT>, <#YearRain24HourHD> - <#r24hourH>, <#Tr24hourH> - - Existing web tags updated: - <#newrecord> - <#RainRecordSet> - - Note: It is not currently possible to edit these records via the built-in records editor -- New: You can now use comment lines that start with a # character within sections in .ini files - - - -3.15.3 - b3173 -—————————————— -- Fix: Broken start-up ping in 3.15.2 - doh! -- Fix: Ecowitt historic catch-up when expected data is missing -- Fix: Disabling the Third Party HTTP Seconds upload no longer requires a restart of CMX - -- Change: Ecowitt historic catch-up now applies a 5 minute offset to the received data - - - -3.15.2 - b3171 -—————————————— -- Fix: Broken start-up ping in 3.15.1 - - -3.15.1 - b3170 -—————————————— -- Fix: Changes to the initial ping delay -- Fix: Remove duplicated Records Set Timeout setting in Station Settings/General/Advanced -- Fix: Some HTTP Ecowitt stations not sending yearlyrainin - try and use totalrainin for these - -- Change: Tweak to the ET calculation -- Change: Improved WeatherLink.com status message logging -- Change: The Ecowitt GW1000 station has been renamed to "Ecowitt Local API" to better reflect the applicability to a range of devices - - The Ecowitt local API remains the preferred method of connection over HTTP (Ecowitt) - - - -3.15.0 - b3169 -—————————————— -- Fix: Prevent real time processing occurring before first data has been received -- Fix: Davis WLL: Add missing decode of THSW from current data -- Fix: Daily high humidex time being logged as high apparent temp time - -- Change: Leaf wetness web tags <#LeafWetness[1-8]> now accept the rc and dp parameters -- Change: Davis WLL: Now fetches temperature data every 10 seconds instead of 60 seconds - -- New: Adds experimental support for Ecowitt stations (GW1000 & HTTP) historic catch-up -- New: HTTP (Ecowitt) station: adds support for WS990 battery state decoding - -- New: Additional web tag for annual ET total <#AnnualET> - - - -3.14.2 - b3162 -—————————————— -- Fix: Dayfile viewer/editor header for log date changed year format to correct value = "dd/mm/yy" -- Fix: HTTP stations now ignore any inbound data until any required pre-processing is complete -- Fix: Removal of old log files from the /MXdiags folder now ignores log files created by other utilities -- Fix: Davis WLL: Adds missing health data decode for soil/leaf transmitters, and adds SuperCap voltage for Vue transmitters -- Fix: GW1000 & HTTP Ecowitt: Wind speed handling now consistent across both protocols -- Fix: Davis VP2: Fix USB/Serial stations stopping polling when connection is temporarily lost - -- Change: Davis WLL: Davis leaf wetness sensors now log values as decimals when reporting - -- New: Third party uploads to WOW can now include soil temperature from any chosen sensor -- New: Adds additional units to the JSON data files - "windrun", "soilmoisture", "co2", "leafwet", and "aq" - - -3.14.1 - b3160 -—————————————— -- Fix: Davis WLL stations recording zero values if the WLL sends "null" for a value -- Fix: Dashboard Alarm settings screen now correctly shows the Sound Enabled flags -- Fix: Dashboard Historic charts nor correctly removes the Temperature Sum chart - -- Change: Ecowitt GW1000 discovery mechanism updated - -- New: The Solar calculation now allows you to input different transparency values for July and December. MX uses a cosine interpolation between the two values - There are changes to cumulus.ini to accommodate this. - [Solar] - RStransfactor=0.8 Removed - BrasTurbidity=2.0 Removed - RStransfactorJul=0.8 New - RStransfactorDec=0.8 New - BrasTurbidityJul=2.0 New - BrasTurbidityDec=2.0 New - - - -3.14.0 - b3159 -—————————————— -- Fix: Uncaught exception in SFTP Interval uploads -- Fix: Endless loop during end of day processing if temperature sensor is missing -- Fix: GW1000, HTTP Ecowitt and HTTP Ambient stations are no longer forced to disable the "Use Speed for Avg" setting -- Fix: WMR928 Station: Crash opening invalid COM port now handled correctly - -- Change: Cumulus MX now uses .Net Framework 4.8 (previously 4.5.2) -- Change: Now supports TLS1.3 for HTTPS/FTPS/MQTT/Email - dependent on OS support and enabled -- Change: MySQL server connection check amended to attempt to work around issues with some servers -- Change: The data stopped, and sensor contact lost alarms are now enabled by default for new installs (both with 1 hour latch, and 2 event trigger) -- Change: The records editors and records pages in the dashboard now display dashes for records yet to be set -- Change: The monthly log file and extra log file file editors now accept a date range rather than showing a whole month at a time - The date range can span a month end - These editors now also have a search field which you can use to search for specific values or times. Only matching lines will be displayed - - It is up to you to be sensible on restricting the date range to the performance of your CMX computer - - A fast machine with SSD drives will cope with much larger ranges than a Raspberry Pi zero for instance! - - -3.13.8 - b3154 -—————————————— -- Fix: Error in GW1000 LiveData decoding - - -3.13.7 - b3153 -—————————————— -- Fix: Davis stations not retrieving 10 min Gust values from LOOP2 packets -- Fix: HTTP Ecowitt stations not decoding user temp sensors (HTTP Ecowitt Extra did) -- Fix: Culture Option to remove spaces from dates - -- Change: Writes to log files are now retried twice if they fail the first time - -- New: adds support for Ecowitt WH90 sensor battery status - - - -3.13.6 - b3152 -—————————————— -- Fix: HTTP Ecowitt decode of leaf wetness sensors again -- Fix: HTTP Ecowitt/Ambient - Remove soil temperature from extra sensor settings -- Fix: Davis VP2 station sending zero values for soil temp instead of 255 when no sensor present -- Fix: Long standing bug in Davis VP2/Vue & WLL stations where the day rollover during catch-up was performed after the monthly log entry rather than before -- Fix: Small tweaks to the ET calculation -- Fix: Davis VP2 console uses 16 bit signed value for Total Packets Received - compensate for it overflowing in CMX - -- New: Program Settings now allows you to over-ride the default date separator string for your locale. This will be useful for those locales that under - Mono now use a date separator such as ". " and you want it to use "." - -- Change: Cumulus will now still parse the dates/times in dayfile.txt and monthly logs even if the locale changes -- Change: FO Station will now ignore UV-I values greater than 16 - -- Updated libraries - MailKit - MimeKit - MySqlConnector - - - -3.13.5 - b3151 -—————————————— -WITHDRWAN due to issues with FluentFTP and Mono when connecting to FTP servers that allow newer EC encryption algorithms - - -3.13.4 - b3149 -—————————————— -- Fix: Dayfile editor MySQL update statement failing and knock on effects -- Fix: Davis AirLink sensors now honours the disabling of auto-discovery, and improve auto-discovery logic -- Fix: Davis WLL station now honours the disabling of auto-discovery -- Fix: HTTP Extra sensors were over-riding the station manufacturer causing various side effects -- Fix: Extra sensor logging is now only automatically enabled for HTTP sensors if the selected sensors are actually logged -- Fix: HTTP Ecowitt station decode of leaf wetness sensors values and battery data - -- New: Adds an estimated evapotranspiration value (ET) for stations that have a solar irradiation sensor and do not already supply this value - The only currently supported station provides an ET value are Davis VP2/WLL stations with solar sensors - The Cumulus calculation uses the standard grass reference crop - -- Change: Instead of 'Cumulus pressure names' being implicitly set internally for Davis stations and having to be explicitly set for some other stations - Davis VP2/WLL and some! other stations will now explicitly set this option for you - - - -3.13.3 - b3148 -—————————————— -- Fix: FineOffset station historic catch-up is now aligned to the correct logger intervals -- Fix: Rain counter reset incorrectly detected if a day has had more than 50 mm of rainfall -- Fix: Realtime FTP would not recover after a connection failure - - -3.13.2 - b3147 -—————————————— -- Fix: HTTP Station Ecowitt - not decoding all CO₂ sensor values correctly, again! - -- Change: The CumulusMX distribution zip no longer contains the required empty folders /backup, /data - These folders will now be created on start-up if they are missing - - -3.13.1 - b3146 -—————————————— -- Fix: HTTP Station Ecowitt - not decoding all CO₂ sensor values correctly -- Fix: AirLink crash when used as a standalone station -- Fix: NOAA settings normal precipitation value for July being saved as June's value - -- New: Ecowitt, adds support for WN34S/WN34L soil/water temperature sensors as User Temps as per GW1000 -- New: The default web site index page now shows the Moon illumination percentage - -- Change: On start-up Cumulus now always checks for other running instances and reports if one is found, but only aborts if 'Stop second instance' is enabled - -- Summary of Cumulus.ini changes: - >> NEW >> - [GW1000] - ExtraSensorUseUserTemp=1 - - -3.13.0 - b3145 -—————————————— -- Fix: Realtime FTP, now correctly clears the RT FTP in progress flag -- Fix: Station Settings, Units, Advanced not being saved correctly -- Fix: The Log editors (dayfile, data, extradata), now handle returned errors correctly -- Fix: Rain not being counted with really small bucket sizes - -- New: Adds support the WeatherFlow Tempest weather stations (contributed by Doug Summersgill) -- New: HTTP Ecowitt: Adds GW1000 Firmware Version extraction to both main station and extra sensors - -- Change: Davis WLL, low battery warning for WLL device changed from 5.2 to 5.4V (1.35V per cell) -- Change: Ecowitt battery decoding nightmare continues and evolves as we discover what they really send as opposed to what they say they send! -- Change: Monthly NOAA reports now stop at the first day with an error in either the dayfile, or the monthly log file. Previously it only stopped at the first dayfile error -- Change: Internet Settings now works better when enabling/disabling FTP globally and in the interval settings -- Change: Ecowitt GW1000 now detects WS80 ultrasonic wind sensors and changes the update rate to 4 seconds (WS80 updates every 4.75 secs) - -- Updated libraries - - MailKit - - MimeKit - - ServiceStack.Text - - -3.12.1 - b3143 -—————————————— -- Fix: Issues with the temperatures on the Gauges page (and adds Feels like) -- Fix: Hourly rain value when the rain counter is reset -- Fix: Limit real time MySQL inserts to 1 minute was not saving correctly -- Fix: Alarm threshold counts not being saved to Cumulus.ini -- Fix: Trend values being set to invalid values on new installs -- Fix: GW1000 station, incorrect decoding of WH51 battery status -- Fix: Davis WLL ET value for the day being reset to zero on MX restart - -- New: Experimental - Add HTTP Station type for Ambient consoles or WH2600/Observer IP - - Set-up your Customized Server in the AWnet app to: - Protocol : Ambient Weather - Server : - Path : /station/ambient? - Port : 8998 (or whatever you have configured CMX for) - Interval : 20 (seconds) -- New: Experimental - You can now add extra sensors to your existing station using an Ambient console or WH2600/Observer IP. - - Set-up your Customized Server in the AWnet app to: - Protocol : Ambient Weather - Server : - Path : /station/ambientextra? - Port : 8998 (or whatever you have configured CMX for) - Interval : 20 (seconds) -- New: Adds configuration help for the HTTP Upload stations (Ecowitt, Ambient, Wunderground) to the Wizard, Station Settings, and Extra Sensors dashboard pages -- New: Implements Alarm threshold counts for Sensor Contact Lost, Data Connection Lost, Low Batteries - -- Change: Removes the restriction that you could not add Ecowitt Extra sensors to an Ecowitt main station -- Change: The Calibration Settings page has been re-organised -- Change: Settings screen more compact for mobile screens, added Aria screen reader attributes to Internet Settings - -- Summary of Cumulus.ini changes: - >> NEW >> - [Ambient] - ExtraSensorDataEnabled=0 - ExtraSensorUseSolar=1 - ExtraSensorUseUv=1 - ExtraSensorUseTempHum=1 - ExtraSensorUseSoilTemp=1 - ExtraSensorUseSoilMoist=1 - ExtraSensorUseAQI=1 - ExtraSensorUseCo2=1 - ExtraSensorUseLightning=1 - ExtraSensorUseLeak=1 - - -3.12.0 - b3141 -—————————————— -- Fix: The Realtime FTP - particularly SFTP - reconnection code has been rewritten to make it more robust -- Fix: SteelSeries gauges data mouseovers not working on the dashboard interface -- Fix: Davis WLL, bug in Chill Hours calculation during archive data catch-up. Each interval increment was 60x larger than it should be -- Fix: Davis WLL, add missing Sunshine hours calculation to historic catch-up -- Fix: Davis WLL, now correctly handles null values in historic data during catch-up -- Fix: GW1000 station, 0.1mm rain tippers would take 3 tips in an interval to register "last rained" when using Inches as the rain unit -- Fix: GW1000 station, adds missing low battery alarm -- Fix: GW1000 station, user temperatures above channel 1 were not being assigned to the correct channel -- Fix: Instromet Increment Logger Pointer was not saving correctly from the Station Settings screen -- Fix: LowTempAlarmSoundFile being written to incorrect Cumulus.ini entry -- Fix: Start-up PING now attempts to catch 'hung' responses and terminates them -- Fix: Start-up delay is now applied before the start-up PING as originally intended -- Fix: Broken 'Stop second instance' code - - -- New: Adds a First Time Setup Wizard to the settings menu. First time users (i.e. no pre-existing Cumulus.ini) are asked to run this from prompts in the console. -- New: Adds a HTTP upload station type using WUnderground protocol - - Set-up your Customized Server in WSView to: - Protocol : Wunderground - Server : - Path : /station/wunderground? - Station Id : 1 (can be anything) - Key : 1 (can be anything) - Port : 8998 (or whatever you have configured CMX for) - Interval : 20 (seconds) -- New: Adds a HTTP upload station type using Ecowitt protocol - - Set-up your Customized Server in WSView to: - Protocol : Ecowitt - Server : - Path : /station/ecowitt - Port : 8998 (or whatever you have configured CMX for) - Interval : 20 (seconds) -- New: You can now add extra sensors to your existing station using an Ecowitt GW1000 (or compatible console). - - Set-up your Customized Server in WSView to: - Protocol : Ecowitt - Server : - Path : /station/ecowittextra - Port : 8998 (or whatever you have configured CMX for) - Interval : 20 (seconds) -- New: Adds more "advanced" options to the GUI configuration: - - Max wind speed in Station > Common Options > Advanced options - - Record set timeout in Station > Common Options > Advanced options - - Snow depth hour in Station > Common Options > Advanced options - - Rain day threshold in Station > Common Options > Advanced options - - List web tags in Program Settings > Program General Options -- New: Adds Cloud base unit to Units settings -- New: Realtime and Web Interval settings changed via the dashboard now have immediate effect, a restart is no longer required. -- New: Adds Lightning distance and last strike time to Today.ini to preserve the values across CMX runs when the GW-1000 has 'forgotten' them - [Lightning] - Distance=-1 - LastStrike=0001-01-01T00:00:00 -- New: Adds a global Enable/Disable switch to Internet FTP settings, the value for this on the first run of this version of CMX will be inferred from the other settings. - From this release onwards, this switch can be used to enable/disable all FTP activity. It does not affect the Interval and real time interval settings. - Disabling this option also disables all the relevant FTP settings in the following sub-sections on the page -- New: Adds an option to Internet settings for "Local Copy". This enables the ability to create all the standard web site files, and copy them on the local file - system instead (or as well as) FTPing them. Useful for installs that run on the same host as the web server. -- New: Two new web tags for sunshine hours - <#SunshineHoursMonth> <#SunshineHoursYear> - - When used without parameters these return the total sunshine hours for this month so far, and this year so far. - - The Month tag accepts three parameters: - y=nnnn & m=nn - use these to specify a specific month you want the total for. Example - <#SunshineHoursMonth y=2021 m=1> for the January 2021 total - r=-nn - use this parameter to specify a relative month, -1 = last month, -2 = month before last etc. Example <#SunshineHoursMonth r=-1> for last month's total - - The Year tag accepts two parameters: - y=nnnn - specify the year you want the total. Example <#SunshineHoursYear y=2020> - r=-nn - specify a relative year, -1 = last year etc. Example <#SunshineHoursYear r=-2> total for the year before last - - The tags also accept the usual rc=y and dp=n parameters -- New: Add New Record timeout value to Station Settings > General > Advanced -- New: Adds Chill hours to yesterday.ini, and adds a new web tag for chill hours total yesterday - <#Ychillhours> -- New: Adds the Chill Hours configuration values to the Station Settings screen -- New: Add new web tags for average wind speed today and yesterday - <#windAvg>, <#windAvgY> -- New: Adds the ability to update the corresponding entries in the Dayfile and Monthly MySQL tables when using the log file editors in the dashboard - - You can enable/disable this feature via a setting in MySQL settings - - It only applies to Edit, Delete does not delete the entry in MySQL -- New: Catch-up from a station logger with (standard) MySQL enabled, now also adds each archive entry to the Realtime table -- New: Adds the ability to buffer failed MySQL commands until the MySQL server becomes available again, or Cumulus MX is restarted - when they will be lost - - Enabled via an option in MySQL Settings - - Note: Whilst Realtime updates are buffered, the uploading of failed queries is only performed by the Log updates -- New: FineOffset stations now log the progress of archive data loading to the console -- New: NOAA Reports now have an option to calculate the mean temperatures using the traditional method (max + min) / 2 - - Note: This does not apply to heating/cooling degree days, they will still use the integrated method -- New: Adds ET for Davis WLL stations with Pro subscriptions - - Note: Your solar sensor MUST be connected to your primary ISS for this to work, otherwise weatherlink.com does not calculate ET - - -- Change: Cumulus MX now uses a persistent database to store the recent 1 minute data - - This means that charts, recent webtags, and internal calculations for trends and periodic values will be more accurate after a restart - - If Cumulus MX is offline for a prolonged period, data for that offline period will obviously still be at the station logging interval resolution -- Change: MQTT now allows multiple topics for both Update and Interval data. - - The format of the MQTT template files has changed to accommodate this - - The template must now be formatted as JSON, however the payload data for each topic can still be in whatever format you like so long as you construct - a valid string - i.e. escape quotation marks - - The default interval template (the update template is similar) now looks like this... - {"topics": [ - { - "topic": "CumulusMX/Interval", - "data": "{\"time\":\"<#timehhmmss>\",\"temp\":<#temp rc=y>,\"humidity\":<#hum>,\"wgust\":<#wgust rc=y>}", - "retain": false - } - ]} - Where the topic name is "CumulusMX/Interval", and the payload is the string "{\"time\":\"<#timehhmmss>\",\"te...rc=y>}" which is formatted as JSON text - - To create a template with multiple topics, use this format... - {"topics": [ - {"topic": "MyName/Topic1", "data": "\"<#timehhmmss>\",<#temp rc=y>", "retain": false}, - {"topic": "MyName/Topic2", "data": "<#hum>", "retain": true} - ]} - Where Topic1 is formatted as CSV and is not retained on the server, and Topic2 as XML and is retained - - The Dashboard Internet > MQTT settings have been updated to reflect these changes -- Change: Moves the FTP Rename/Delete/UTF-8 Encode settings from Internet Settings|Web Settings|General, to Internet Settings|Web Site|General -- Change: Moves the Forum URL and Webcam URL settings from Internet Settings|Web Site to Internet Settings|Miscellaneous -- Change [Internal]: The Dashboard web pages now all use a common menu.js script to render the menus - it saves me a lot of typing when anything changes! -- Change: The Dashboard settings pages now load the forms JSON as static files rather than via the API. -- Change: The console output on start-up has been changed slightly to show actual IP addresses in URLs, and I have added some colour. - - You now also get different messages if no Cumulus.ini file is detected, directing users to run the first time setup wizard -- Change: The default web site NOAA reports page is now accessible and removes the mouse-over month selection -- Change: When configuring the Davis WLL station, if any extra sensors are enabled, then Extra Logging is also automatically enabled. -- Change: Default unit values for new installs changed for Wind, Pressure and Altitude... - m/s -> km/h - mbar -> hPa - feet -> metres -- Change: When reading the Cumulus.ini file at start-up, if settings are migrated, or invalid entries corrected, Cumulus.ini is now re-written. - That is re-written as created afresh, not updated in place. Old entries are removed, and the file order resequenced. Thus you cannot then - use the new Cumulus.ini on old versions of Cumulus, you will have to use backups. -- Change: Cosmetic changes to most Settings pages, and Records editors -- Change: Some libraries updated: - - Email: MailKit - - MQTT: MQTTnet - - MySQL: MySqlConnector - - JSON: ServiceStack.Text - - -- Summary of Cumulus.ini changes: - >> NEW >> - [FTP site] - Enabled=1 - EnableLocalCopy=0 - LocalCopyFolder= - CopyMoonImage=0 - RealtimeTxtCopy=0 - RealtimeGaugesTxtCopy=0 - Copy-websitedata=0 - Copy-wxnow=0 - Copy-graphconfig=0 - Copy-availabledata=0 - Copy-tempdata=0 - Copy-pressdata=0 - Copy-winddata=0 - Copy-wdirdata=0 - Copy-humdata=0 - Copy-raindata=0 - Copy-dailyrain=0 - Copy-dailytemp=0 - Copy-solardata=0 - Copy-sunhours=0 - Copy-airquality=0 - Copy-alldailytempdata=0 - Copy-alldailypressdata=0 - Copy-alldailywinddata=0 - Copy-alldailyhumdata=0 - Copy-alldailyraindata=0 - Copy-alldailysolardata=0 - Copy-alldailydegdaydata=0 - Copy-alltempsumdata=0 - - [MySQL] - UpdateOnEdit=1 - BufferOnFailure=0 - - [GW1000] - ExtraSensorDataEnabled=0 - ExtraSensorUseSolar=1 - ExtraSensorUseUv=1 - ExtraSensorUseTempHum=1 - ExtraSensorUseSoilTemp=1 - ExtraSensorUseSoilMoist=1 - ExtraSensorUseLeafWet=1 - ExtraSensorUseAQI=1 - ExtraSensorUseCo2=1 - ExtraSensorUseLightning=1 - ExtraSensorUseLeak=1 - - [Graphs] - MoonImageCopyDest=images/moon.png - - [NOAA] - AutoCopy=0 - CopyDirectory= - UseMinMaxAvg=0 - - >> NO LONGER USED >> - [FTP site] - AutoUpdate=0 // Deprecated, now read-only for migration purposes - - [MQTT] - UpdateTopic=CumulusMX/DataUpdate - UpdateRetained=0 - IntervalTopic=CumulusMX/interval - IntervalRetained=0 - - - -3.11.4 - b3133 -—————————————— -- Fix: MySQL crash - "Adding the specified count to the semaphore would cause it to exceed its maximum count" - - - -3.11.3 - b3132 -—————————————— -- Fix: HTTP Alarm alerting for WUnderground successful uploads -- Fix: Davis WLL setting dewpoint to zero on processing the first history record if Cumulus Calculates Dewpoint is selected -- Fix: FTP Now did not always update all the Graph JSON files and flag them for FTP - - - -3.11.2 - b3131 -—————————————— -- Fix: Send email failing if email logging is not enabled -- Fix: Solar Rad colour now displays as selected on first load in the Selectachart on the dashboard and default website -- Fix: Handlebars JS script updated due to severe security risk on previous version -- Fix: GW1000 for 433, 868, 915 MHz variants -- Fix: Bug in MySQL catch-up - -- Change: Setting of ALL logging options via the GUI is now "sticky" - i.e. preserved across runs of Cumulus MX -- Change: Many of the main settings screens now alert to to the first invalid settings in a message, and the tree containing - the error is highlighted in red: - - Program Settings - - Station Settings - - Internet Settings - - Third Party Settings - - Extra Sensor Settings - - Calibration Settings - - NOAA Settings - - MySQL Settings -- Change: If any of the base units (wind, temp, rain, pressure) are changed via Station Settings (for instance during the initial configuration), then all the - thresholds and base values associated with that unit are reset to the defaults for the new unit -- Change: The main dashboard page now only shows alarm indicators for enabled alarms -- Change: Alarm Settings page now has added screen reader attributes (not visible on the page) -- Change: All HighCharts based pages (Dashboard and default web site) now use the latest stable release of HighCharts - -- New: Implements two new alarms for HTTP uploads failing, and MySQL uploads failing - - These plus Data Spike, have an additional setting via Cumulus.ini (for now) to set a trigger threshold. - This defaults to 1, so the alarm will trigger immediately. If set to 5, then it will take 5 trigger events happen within the Latch time to set the alarm. - The idea is to prevent an occasional error from triggering the alarm - Cumulus.ini settings: - [Alarms] - SpikeAlarmTriggerCount=1 - HttpUploadAlarmTriggerCount=1 - MySqlUploadAlarmTriggerCount=1 -- New: Web tags - <#HttpUploadAlarm>, <#MySqlUploadAlarm> -- New: Data Stopped, Data Spike,HTTP upload, MySQL upload emails now also report the error that triggered the alarm in the email text -- New: Adds two new variables for Extra Web Files - & to upload the last year and month reports. It only really makes sense to use these with the EOD option -- New: You can now limit the real time MySQL table inserts to once a minute. Useful if you run a short real time interval (say 5 seconds), and do not want to - flood your MySQL real time table with many rows that hardly change. - - - -3.11.1 - b3130 -—————————————— -- Fix: Fix Test email logging "success" on failure -- Fix: Rollback FluentFTP package to previous 32.3.1 version due to crashes in Mono with newer versions - -- New: Pressure change in last 3 hours web tag - <#PressChangeLast3Hours> -- New: Enable additional Accessibility feature - configured via either Station Settings or Program Settings - The following pages now have some enhanced accessibility features: - - Program Settings - - Station Settings - - Internet Settings - - Third Party Settings - - Extra Sensor Settings - - Calibration Settings - - NOAA Settings - - MySQL Settings - Adds a new entry in Cumulus.ini - - - -3.11.0 - b3129 -—————————————— -- Fix: Remove THW Index in /web/websitedataT.json, and default web site index.htm -- Fix: OpenWeatherMap - create new station was not forcing the use of dot decimals for lat/long values -- Fix: WeatherCloud - date/time of reading should be UTC -- Fix: End of day backup now always runs at rollover -- Fix: FineOffset stations were not ignoring invalid wind speeds on historic catch up -- Fix: FineOffset stations historic catch up was failing when it reached a logger memory wrap around -- Fix: FineOffset stations historic catch up is now limited to the number of logger entries the console says are recorded -- Fix: FineOffset synchronise reads process improvements (attempt) -- Fix: Catch some errors that could occur in the start-up PING process -- Fix: MySQL connections that use encryption should now be supported (MySQL 8.0+ default) -- Fix: GW1000 System Information now correctly decodes 915 and 920MHz devices -- Fix: GW1000 occasional crash when an unexpected response was received -- Fix: All settings screens error handling messages now no longer display "[object object]" - -- New: Adds support for a THW Index calculation. This value is now available for all station types via the web tag <#THWindex> -- New: Adds support for Windguru uploads -- New: Adds Email support for Alarms - - Configured via Internet Settings (email server config), and Alarm Settings (from/to addresses for alarms, and a Test Email function) - - Creates a new section in Cumulus.ini: - [SMTP] - ServerName= - Port=587 - SSLOption=1 - RequiresAuthentication=0 - User= - Password= - Logging=0 - - Adds to the Alarms section in Cumulus.ini - [Alarms] - xxxxxAlarmEmail=0 - FromEmail= - DestEmail= - - Creates a new section in strings.ini, with email text - [AlarmEmails] -- New: WeatherCloud now supports uploading of Air Quality, Soil Moisture, and Leaf Wetness -- New: Adds support for two sets of Growing Degree Days data - - Configured via Station Settings|Growing Degree Days, visibility via Station Settings|Graphs|Data Series Visibility|Degree Days -- New: Adds support for Temperature Sum - annual running total of daily average temperatures - - Configured via Station Settings|Temperature Sum, visibility via Station Settings|Graphs|Data Series Visibility|Temperature Data -- New: The graphs pages - both dashboard and default web site - now set a page hash value depending on the graph being shown. - Benefit = you can now link directly to a particular graph to show on page load -- New: FineOffset stations now report a warning if the console logger interval does match the Cumulus MX interval -- New: FineOffset experimental feature to set the console logger interval to match Cumulus logging interval -- New: JavaScript encoded web tags. These tags were previously only available as HTML entity encoded strings: - <#latitudeJsEnc>, <#longitudeJsEnc> - <#locationJsEnc>, <#longlocationJsEnc> - <#forecastJsEnc>, <#cumulusforecastJsEnc>, <#wsforecastJsEnc> - <#currcondJsEnc>, -- New: Web tags with no HTML entity encoding: - <#tempunitnoenc>, <#altitudenoenc> - -- Changed: Moved FTP Logging option from Internet Settings to Program Settings to collect all the logging options in one place -- Changed: A Data Stopped state now stops all logging, MySQL and web activity -- Changed: Updates to various library components... - - Updated to latest versions: - - FTP: FluentFTP - - JSON: ServiceStack.Text - - Pointers: System.Runtime.CompilerServices.Unsafe - - MQTT: MQTTnet - - Removed/Added - - MySQL: Removed Devart, replaced with MySqlConnector - - MailKit: Added along with supporting packages MimeKit, and Portable.BouncyCastle -- Changed: FineOffset stations now attempt to reconnect the USB after a data stopped event. These stations now also keep retrying to connect to the station at start-up -- Changed: The MySQL real time table data retention settings reworked. -- Changed: Moved all third party web uploads from Internet Settings page to their own Third Party Settings page -- Changed: Internal optimisations -- Changed: JQuery library updated to v3.6.0 for both the Dashboard interface, and the default web site - - - -3.10.5 - b3122 -—————————————— -- Fix: Comma decimal issues with OpenWeatherMap and other third party uploads -- Fix: Occasional GW1000 crash on GetSensorIdsNew response timeout -- Fix: GW1000, WH40 rain gauge, removed battery status check as it does not send this information -- Fix: Adds missing WeatherCloud interval setting to Internet Settings -- Fix: Default web site Monthly NOAA reports before the year 2010 - -- New: Now determines the Ecowitt GW1000 device main sensor type (WH65/WH24) and prints system information to the log file -- New: Adds the free text station model to the Station settings page -- New: Adds a cache buster to default web site webpagedata.json downloads -- New: Adds optional UV index to Now, Today, Yesterday pages of the default web site -- New: All decimal web tag values now accept a dp=N and tc=y overrides for the number of decimal places -- New: Adds option to upload AQ data to WeatherCloud - - -3.10.4 - b3121 -—————————————— -- Fix: Issue introduced in v3.10.3 decoding Ecowitt live data -- Fix: Catch real time FTP updates that have hung for more than 5 minutes and restart them. - - - -3.10.3 - b3120 -—————————————— -- Fix: Problem with setpagedata.js in "/legacy-webfiles" -- Fix: Ecowitt GW1000 and clone auto-discovery extended to WH2650 and Ambient clones -- Fix: Ecowitt auto-discovery broken after a recent Ecowitt firmware update -- Fix: Ecowitt WH45 battery status on firmware levels below 1.6.5 -- Fix: Blake-Larsen sunshine recorder, the SRsunshine.dat file was using user locale decode, it should always use dot decimal -- Fix: MQTT only read the Update template file at start-up or if the filename was changed. It now reads the file at every update -- Fix: Davis WLL was not calculating chill hours and heating/cooling degree days on catch-up -- Fix: Add missing COM port setting for OS WM918/WMR928 stations - -- New: Adds Records Began Date to - Station Settings|General|Advanced -- New: Adds support for GW1000 firmware 1.6.6 -- New: Default web site - You can now use the "data-cmxdata" attribute on any HTML element, not just spans. BUT note, ALL the innerHTML will get replaced with the JSON data. -- New: Two new web tags that HTML encode the station description strings - <#locationenc>, <#longlocationenc> -- New: Adds a NOAA Reports page to the default web site -- New: Adds support for the EcoWitt WH35 8 channel leaf wetness sensor - - Note that only channels 1-4 will be displayed on the dashboard - - Extends the leaf wetness web tags with <#LeafWetness5> to <#LeafWetness8> added - - samplestrings.ini adds leaf wetness captions 5-8 -- New: Experimental - Enables Battery Low alarm for WMR100/WMR928 stations - -- Changed: The Default web site menu system has been rewritten. It is now all defined in the file menu.js - - -3.10.2 - b3117 -—————————————— -- Fix: Improve the AWEKAS fall back for upload interval to go right back to 5 minutes in two stages -- Fix: Occasional corrupt files output that have processed web tags in them -- Fix: Error creating the NOAA Year report for some people -- Fix: Missing station location on gauges.htm web site page - -- New: Default website now removes the "Current Conditions" element if the value is blank -- New: Default web site now auto updates the index.htm and today.htm pages every minute - -- Change: The current conditions is now HTML encoded in case it contains illegal characters - -- Updated default web site files - \web\websitedataT.json - \webfiles\index.htm - \webfiles\gauges.htm - \webfiles\js\setpagedata.js - - - -3.10.1 - b3116 -—————————————— -- Fix: Bug in temperature conversions introduced in v3.10.0 - - -3.10.0 - b3115 -—————————————— -- Fix: Catch error creating System Uptime counter on Windows -- Fix: The Local web server is now brought up before initialising the station. This allows you to correct a misconfigured station without resorting to editing the Cumulus.ini file. -- Fix: Diary Editor creating entries on the wrong day, and revamp the interface a bit -- Fix: WLL day average temp stats on historic catch-up -- Fix: GW1000 auto-discovery was triggering an erroneous IP address change -- Fix: Cumulus MX shutdown when running as a system service is now orderly -- Fix: The start-up ping now refreshes the DNS cache before every re-try to avoid using null entries cached before the internet comes up - -- New: Brand new default web site template courtesy of Neil Thomas. The original "legacy" web site is still included, but it has been moved to the /webfiles-legacy folder. - - The new web site is now data file driven as opposed to all pages being processed and uploaded. The legacy web site has also been updated to use this method. -- New: The previous Console log file is now copied to an "-old" file on start-up -- New: For Davis WLL stations using weatherlink.com, Cumulus now checks and reports the operational status of weatherlink.com on start-up and if an error occurs accessing the service -- New: Two web tags <#forumurl> and <#webcamurl>, which just return the respective URLs rather than the pre-canned HTML of <#forum> and <#webcam> -- New: The start of a Display Options section under Station, which controls what data is displayed on the default web site, implementing some of the Cumulus 1 options - - New web tags for this: <#Option_useApparent>, <#Option_showSolar>, <#Option_showUV> - -- Change: All the settings screens revamped, reorganised and extended. - - Many of the settings are now context sensitive, only showing items relevant to your station and configuration. - - Most of the previously config file "read-only" settings are now available in an Advanced section relevant to the configuration item. These settings are now read/write. - - Many new Cumulus.ini configuration entries, and some now depreciated. - - Virtually all the standard files that can be generated can now be controlled for enabling/disabling generation and FTP transfer independently. - - Added more graph data series controls. -- Change: The two graph config files availabledata.json and graphconfig.json are now only uploaded on program start-up and when the station config is changed. -- Change: Dayfile, Monthly Log, and Extra log file editors now have a selectable page length, and a goto-page feature -- Change: The default web site is now driven by a single data file (plus realtimegauges.txt), rather than every page being updated and uploaded each interval. -- Change: The various charting pages now hide buttons for graphs that do not contain any data - both on the dashboard and default web site. -- Change: Creation of the wxnow.txt file is now disabled by default for new installs -- Change: Clock sync (Davis VP2 & Instromet) now occurs at 2 minutes past the hour selected -- Change: Davis VP/VP2/Vue ReadReceptionStats now defaults to enabled for new installs -- Change: The default output file format is now UTF-8 for new installs - - -3.9.7 - b3107 -————————————— -- Fix: Unhandled exception in ProcessTemplateFile if cumulus cannot write to the output file -- Fix: Davis IP logger prevents Cumulus exiting if no station is found on the network -- Fix: Ecowitt AQ sensors not loading graph data from the log files on start-up -- Fix: Bug in dayfile parser that was using the time of highest wind speed as time of lowest humidity. - - The dayfile parser now outputs the first field number in a line that has failed to parse along with the field contents -- Fix: The GW1000 retains the previous lightning time/distance if the station resets the values to defaults -- Fix: Now opens the Davis WLL multi-cast listening port in shared mode to allow multiple copies of Cumulus to run on the same computer. - Only tested on Windows, may not work on macOS. - -- New: Adds a Select-a-Chart to the dashboard for recent data so you can plot different data on the same chart -- New: Adds a Select-a-Chart to the default web site for recent data - - Creates a new json data file - availabledata.json - that is uploaded to the remote site -- New: Adds a hot link to the Upgrade alarm on the dashboard -- New: Adds a start-up host ping escape time. Allows Cumulus to continue loading even if no ping response is received for the specified time - -- Change: The default web site gets a CSS change that dynamically alters the "content" width depending on the screen size. On smaller screens the white space either side will reduce in size. -- Change: The graph JSON data files are now always created locally, regardless of the Include Graph Data Files setting. That now setting only controls FTP like the standard files. - - -3.9.6 - b3101 -————————————— -- Fix: NOAA monthly report min temp/rain not formatting in dot decimal correctly -- Fix: Windows console log no longer being created -- Fix: Fix for exit handlers being called before cumulus instance is initialised causing an unhandled exception -- Fix: Airlink auto-discovery/update when you have more than one Airlink device -- Fix: Upgrade & Spike alarm latch hours not being saved/read correctly. You will need to check the values after installing this release. -- Fix: Rework both the interface graphs and default web site graphs JavaScript files -- Fix: SampleStrings.ini changes/fixes. The following entries should not longer be quoted - Davis Forecast strings and CO₂ captions - -- Change: Recent graph hours now defaults for new installs to 72 hours instead of 24 -- Change: All alarm latch hours now default to 24 hours - - -3.9.5 - b3100 -————————————— -- Fix: Alarms being cleared after their defined latch time even if latching was disabled -- Fix: Adds missing dew point calculation for GW1000 extra T/H sensors 1-8 -- Fix: Potential Object reference not set to an instance of an object fault on initial configuration [Goran Devic] -- Fix: Failed connections to some SFTP servers - Renchi.SshNet library reverted to older version -- Fix: Improved accessibility of the "Extra web files" page -- Fix: Tweak to Sunshine hours counter reset logic (primarily for Imet stations during a power cycle) -- Fix: Davis Console battery low threshold changed from 4.0 V to 3.5 V -- Fix: Rework station rain counter reset logic. Cope with Davis station start-up on the first day of new rain season which is not January - -- New: More comprehensive support for the Ecowitt WH45 CO₂ sensor... - - Added to Extra Sensors page - - Added to Strings.ini - - Can be selected as the primary AQ sensor - - New web tags: - <#CO2-pm2p5>, <#CO2-pm2p5-24h>, <#CO2-pm10>, <#CO2-pm10-24h>, <#CO2-temp>, <#CO2-hum> - - Added to ExtraLog file, columns for: CO2, CO2Avg, pm2.5, pm2.5Avg, pm10, pm10Avg, Temp, Hum -- New: Improved GW-1000 device auto-discovery, after initial discovery or user input of the IP address it locks onto the MAC address for further changes in IP - If multiple potential devices are found they are now listed and the user must add the appropriate IP into the config and restart Cumulus. - Discovery is now run whenever a Data Stopped condition is detected (and periodically thereafter until the data resumes) - New Cumulus.ini setting: - [GW1000] - MACAddress= -- New: Adds a browser cache time limit of 5 minutes to the Admin interface static files -- New: NOAA reports settings now have an option to force "dot" decimal points, this overrides your locale setting - New Cumulus.ini setting: - [NOAA] - UseDotDecimal=0 -- Tweaks the charts for both the admin interface and the default web site -- Rework of Airlink sensor implementation - no functional changes - - -3.9.4 - b3099 -————————————— -- Fix: WMR100 stations generated empty daily graph data files on start-up -- Fix: System uptime web tag for Linux systems - maybe! -- Fix: Historic charts wind run tooltip suffix -- Fix: High Min Temp and Low Max Temp records editors were not displaying and allowing input of the time -- Fix: This Month/This Year records editors were showing records for the entire day file rather than the specific period -- Fix: GW1000 station dewpoint calculation incorrect on first packet read at start-up - -- New: Settings page: Program Settings. Moved Debug/Data logging options and Prevent Second Instance to here from Station Settings. - Note: These settings still remain in the [Station] section of the Cumulus.ini file for backwards compatibility -- New: Two settings to delay the start-up of Cumulus MX. Both added to the new Program Settings page. - Start-up Host PING: Specify a remote hostname or IP address, Cumulus will wait for a successful PING from this address before continuing - Start-up Delay: Specify a fixed time in seconds to delay the start-up. The delay will be executed after the PING check is one is specified - Start-up Delay Max Uptime: Specify a maximum system uptime in seconds after which the delay will no longer be applied. Set to zero to always delay - New section in Cumulus.ini: - [Program] - StartupPingHost= // Default is no remote host - enter a host name or IP address - StartupDelaySecs=0 // Default is no delay = 0 - StartupDelayMaxUptime=300 // Default is 300 seconds (5 minutes) -- New: FTP Now - Adds a new option to regenerate the daily graph data files (if enabled), and include them in the manually invoked FTP session -- New: Adds support for OpenWeatherMap uploads (and catch-up). Using an existing station if the id is supplied, or your API key only has one station defined. - If no stations are defined CMX automatically creates a new one using your station details and starts uploading to it. - - - -3.9.3 - b3098 -————————————— -- Fix: Records check was triggering for very small value changes in derived values such as feels like temps -- Fix: Catch uncaught exception on SFTP disconnect -- Fix: FTP was inconsistent in empty remote server folder handling for real-time and interval files -- Fix: AirLink was not using the local vs WLL API data correctly -- Fix: EU CAQI AQI index calculations (based on the City/Urban scale) -- Fix: US EPA AQI calculation error of values between 51 and 100 -- Fix: Removes redundant alldailywdirdata.json file from daily uploads -- Fix: Corrects incorrect/invalid JSON being created in the alldailytempdata.json file when the dayfile contains blank entry's for dew point -- Fix: Attempted FTP uploads of files that do not exist caused the FTP session to become non-functional -- Fix: The monthly data log file editor erroring with 404 - -- New: You can now use "" in the extra files local filename to copy/upload the latest AirLink data log file -- New: You can now define your Indoor AirLink as the primary AQ sensor. Note this only applies to the admin interface, its data will not be - uploaded to the default web site -- New: Adds Cumulus MX update available alarm to the dashboard - enabled by default. This also has an associated web tag <#UpgradeAlarm> -- New: If enabled, the daily graph data files are now created at start-up and uploaded at first interval FTP. - Previously you had to wait for the first EOD rollover. -- New: If a FineOffset station is not found, MX now dumps a list of all the USB devices it can see to the MXdiags log - - - -3.9.2 - b3097 -————————————— -- Fix: Change log messages for number of SQL rows affected to Debug only messages -- Fix: AirLink web tags no longer error if the sensor is not enabled - all now return "--" -- Fix: Australia NEPM AQIs to allow values greater than 101 -- Fix: Canada AQHI - still based on PM2.5 3 hour values only -- Fix: Dutch and Belgian AQI calculations -- Fix: Default web site temperature Trends graph yAxis issue -- Fix: Broken logfile, extralogfile, and dayfile editors in b3096 -- Fix: Improve Instromet stations error handling of out of sequence responses -- Fix: Extra files without the EOD flag were not being transferred during the EOD upload period (first after rollover) - - -- Adds Notifications to Alarms - Note: Cumulus notifications only work by default if you run the browser on the same machine as Cumulus and connect using the - url: http://localhost:8998 - In order to get them working on a remote machine you will have to change an advanced flag in your browser. - This is at your own risk as advanced flags are unsupported features in Chrome and Edge. - In Chrome or Edge, open the url - "about:flags" - Find the entry "Insecure origins treated as secure" and enable it - The add the url you use to remotely access the Cumulus dashboard - eg. http://192.168.1.23:8998 - Then click the button to restart your browser. Notifications should now work. -- Adds Latch times to Alarms. You can now specify a period in hours that an alarm will remain active after the event that tripped it has reset. -- The default value for "DavisIncrementPressureDP" is changed from true to false - ** The effect of this is that for Davis VP/VP2 stations, the default number of decimal places used for pressure values changes - from 2 dp (hPa/mb) and 3 dp (kPa, inHg) to 1 dp (hPa/mb) and 2 dp (kPa, inHg) - ** If you wish to retain the previous behaviour, then you MUST add the setting "DavisIncrementPressureDP=1" to your Cumulus.ini file -- Wind Rose is now populated from data files at start-up (last 24 hours) -- New graphs option to show/hide sunshine hours data -- New admin interface historic graphs (and associated API) -- New default web site page to show historic graphs -- Adds Air Quality to Recent Graphs in Admin Interface and default web site - configure a primary AQ sensor in Station.Options -- Adds Air Quality upload to WeatherUnderground. Enable in WeatherUnderGround settings, and configure a primary AQ sensor in Station.Options -- Adds Air Quality upload to AWEKAS. Enable in AWEKAS settings, and configure a primary AQ sensor in Station.Options -- Adds ability to define the number of decimal places used for various weather units via read-only settings in Cumulus.ini. - The full list of existing and new values is: - [Station] - WindSpeedDecimals=1 // Existing entry - WindSpeedAvgDecimals=1 // Existing entry - WindRunDecimals=1 // NEW - SunshineHrsDecimals=1 // Existing entry - PressDecimals= // NEW [hPa/mb=1, inHg=2] - RainDecimals= // NEW [mm=1, in=2] - TempDecimals=1 // NEW - UVDecimals=1 // NEW - AirQualityDecimals=1 // NEW - affects AQI values only, not PM values -- Lots of internal code fettling -- New Cumulus.ini read-only entry to control the data read rate for Instromet stations. The value is in milliseconds, default is 500. - [Station] - ImetReadDelay=500 -- Admin interface tweaks -- Change WUnderground Password field name to Station Key in settings - - -3.9.1 - b3096 -————————————— -- Fix: AirLink - Limit the number of decimal places in the AirLink log file -- Fix: AirLink - The web tags for 1hr average AQI values were picking up the latest reading value in error -- Fix: Send rounded wind speed to Weather Underground/Windy/PWS/WoW if Cumulus "Round Wind Speeds" is set -- Fix: Replaces a bad copy of trendsT.htm that crept into b3095 -- Fix: Bug in wind spike removal that prevented it working -- Fix: Airlink caused premature day rollover processing during catch-up -- Workaround for the Chromium browser CSS bug that adds scroll bars to the settings screens -- Adds a new AirLink Sensors page to the admin interface -- Adds Netherlands LKI, and Belgian BelAQI AQI scales [6=Netherlands=LKI, 7=Belgium-BelAQI] to AirLink devices -- All AirLink AQI web tags now support the tc=y parameter, to properly truncate the AQI to an integer value -- Adds the ability to "latch" all alarms for a user configured number of hours -- Adds a new web tag <#CPUTemp> - only works on Linux, updates once a minute -- Change of JSON library from Newtonsoft to ServiceStack.Text -- Dashboard now updates every 2.5 seconds to match Davis stations (was 3 seconds) - -- Updated files - \AirLinkFileHeader.txt - NEW - \CumulusMX.exe - \MQTTnet.dll - \Newtonsoft.Json.dll - REMOVED - \ServiceStack.Text.dll - NEW - \System.Buffers.dll - NEW - \System.Memory.dll - NEW - \System.Runtime.CompilerServices.Unsafe.dll - NEW - \interface\*.html - ALL html files - \interface\css\cumulus.css - \interface\json\ExtraSensorOptions.json - NEW - \interface\json\ExtraSensorSchema.json - NEW - \interface\json\InternetSchema.json - \interface\json\InternetOptions.json - \interface\js\airlink.js - \interface\js\extrasensorsettings.js - NEW - \interface\alarmsettings.js - \interface\lib\alpaca\alpaca.css - \web\trendsT.htm - - - - -3.9.0 - b3095 -————————————— -- Fix: Running as a Windows service, now correctly reads command line parameters from the registry "ImagePath" field -- Fix: Cumulus forecast would sometimes start with a comma -- Fix: trendsT.htm location formatting -- Adds support for the Davis AirLink air quality monitor. - - Like the WLL it supports auto-discovery of the AirLink IP address, this can be disabled via config - - A new Settings page has been created for "Extra sensors", this is where you can configure the AirLink. - - Logs AirLink data to a new CSV data file "AirLinklog.txt" - see AirLinkFileHeader.txt for field information - - Historic catch-up is supported if the main station is a Davis WLL and the AirLink has been added as node to this station - - Historic catch-up for other stations will be a later release - - Creates a new Cumulus.ini file section - [AirLink] - WLv2ApiKey= - WLv2ApiSecret= - AutoUpdateIpAddress=1 - In-Enabled=0 - In-IPAddress=0.0.0.0 - In-IsNode=0 - In-WLStationId= - Out-Enabled=0 - Out-IPAddress=0.0.0.0 - Out-IsNode=0 - Out-WLStationId= - AQIformula=0 [0=US-EPA, 1=UK-COMEAP, 2=EU-AQI, 3=EU-CAQI, 4=Canada-AQHI, 5=Australia-NEMP] - - Adds new web tags for Davis AirLink - <#AirLinkFirmwareVersionIn>,<#AirLinkWifiRssiIn> - Requires Davis WeatherLink Live Pro subscription - <#AirLinkFirmwareVersionOut>,<#AirLinkWifiRssiOut> - Requires Davis WeatherLink Live Pro subscription - <#AirLinkTempIn>,<#AirLinkHumIn> - <#AirLinkTempOut>,<#AirLinkHumOut> - <#AirLinkPm1In>,<#AirLinkPm1Out> - <#AirLinkPm2p5In>,<#AirLinkPm2p5_1hrIn>,<#AirLinkPm2p5_3hrIn>,<#AirLinkPm2p5_24hrIn>,<#AirLinkPm2p5_NowcastIn> - <#AirLinkPm2p5Out>,<#AirLinkPm2p5_1hrOut>,<#AirLinkPm2p5_3hrOut>,<#AirLinkPm2p5_24hrOut>,<#AirLinkPm2p5_NowcastOut> - <#AirLinkPm10In>,<#AirLinkPm10_1hrIn>,<#AirLinkPm10_3hrIn>,<#AirLinkPm10_24hrIn>,<#AirLinkPm10_NowcastIn> - <#AirLinkPm10Out>,<#AirLinkPm10_1hrOut>,<#AirLinkPm10_3hrOut>,<#AirLinkPm10_24hrOut>,<#AirLinkPm10_NowcastOut> - <#AirLinkAqiPm2p5In>,<#AirLinkAqiPm2p5_1hrIn>,<#AirLinkAqiPm2p5_3hrIn>,<#AirLinkAqiPm2p5_24hrIn>,<#AirLinkAqiPm2p5_NowcastIn> - <#AirLinkAqiPm2p5Out>,<#AirLinkAqiPm2p5_1hrOut>,<#AirLinkAqiPm2p5_3hrOut>,<#AirLinkAqiPm2p5_24hrOut>,<#AirLinkAqiPm2p5_NowcastOut> - <#AirLinkAqiPm10In>,<#AirLinkAqiPm10_1hrIn>,<#AirLinkAqiPm10_3hrIn>,<#AirLinkAqiPm10_24hrIn>,<#AirLinkAqiPm10_NowcastIn> - <#AirLinkAqiPm10Out>,<#AirLinkAqiPm10_1hrOut>,<#AirLinkAqiPm10_3hrOut>,<#AirLinkAqiPm10_24hrOut>,<#AirLinkAqiPm10_NowcastOut> - <#AirLinkPct_1hrIn>,<#AirLinkPct_3hrIn>,<#AirLinkPct_24hrIn><#AirLinkPct_NowcastIn> - <#AirLinkPct_1hrOut>,<#AirLinkPct_3hrOut>,<#AirLinkPct_24hrOut><#AirLinkPct_NowcastOut> -- The Blake-Larsen sunshine recorder, and Hydreon RG-11 rain device configuration options can now be set on the new Extra Sensors page. - There are two new settings for RG-11 devices, if you use these devices you must set these via the configuration page as they will default to disabled. - [Station] - RG11Enabled=0 - RG11Enabled2=0 -- Adds new graph option - Solar Visible. Setting both Solar and UV to hidden means the solar graph data files will be empty. -- CumulusMX no longer re-writes the Cumulus.ini file on shutdown -- Adds the option to set the retained flag for the MQTT feeds - - New Cumulus.ini file settings - [MQTT] - UpdateRetained=0 - IntervalRetained=0 - -- Updated files - \CumulusMX.exe - \AirLinkFileHeader.txt - \interface - lots of changes, all *.html files, /scripts, and /json - \web\trendsT.htm - - -3.8.4 - b3094 -————————————— -- Fix: Unhandled exception in MySQL during the start-up catch-up process -- Fix: Graph buttons not wrapping on default web site trends page -- Fix: Calculating the trend rain rate when a rainfall multiplier is used -- Fix: Program Uptime when running under Mono and uptime > 24 days -- Fix: When running as a Windows service, the service was stopped unintentionally on non-critical system power events -- Fix: Davis VP receptions stats - ReceptionStats, FirmwareVersion and BaroRead functions rewritten -- Adds Ecowitt GW1000 decoding of WH34 battery status -- Adds Ecowitt GW1000 support for CO₂ sensor, no logging yet, but the decoding is there for when these sensors are released. - - New web tags <#CO2>, <#CO2-24h> -- Adds new Linux cumulusmx.service configuration file (thanks to @freddie), this allows you run Cumulus MX as a service under Linux using the - newer systemd control rather than the original init.d script supplied with v3.8.0 - - Edit the cumulusmx.service configuration file and change the path to match your Cumulus MX installation folder - - Copy cumulusmx.service to your /etc/systemd/system/ folder - - Cumulus can then be started and stopped with the following commands: - > systemctl start cumulusmx - > systemctl stop cumulusmx - - Status and restart are available too: - > systemctl restart cumulusmx - > systemctl status cumulusmx - - If you make a change to the unit file, you need to run the following command so that systemd re-reads your file: - > systemctl daemon-reload - - To have Cumulus start on reboot, issue the following command: - > systemctl enable cumulusmx - - Finally, to stop the service running on reboot, use the following: - > systemctl disable cumulusmx -- Adds a new web tag <#ProgramUpTimeMs> that returns the Cumulus run time in milliseconds - -- Updated files - \CumulusMX.exe - \web\trendsT.htm - -- New files - \MXutils\linux\cumulusmx.service - - -3.8.3 - b3093 -————————————— -- Fix: The dew point threshold limit (default value 40C) was incorrectly being applied to installations configured to use Fahrenheit - as if the value were in Fahrenheit. -- Fix: Davis VP/VP2 force 1 minute barometer updates no longer working correctly after v3.8.2 protocol changes. - Note: this setting should only be used for VP stations, or VP2 stations with very old firmware. -- Fix: Last rain date not updating for Davis stations (fixes the mixed unit/bucket size update in 3.8.2) -- Adds two new Cumulus.ini file settings to control the number of decimal places used for wind speeds. These override the built-in defaults. - These settings are read-only, they can to be manually added ONLY if you need to use them. Add either or both settings as required. - [Station] - WindSpeedDecimals=[0|1|2] - WindSpeedAvgDecimals=[0|1|2] - -- Updated files - \CumulusMX.exe - - -3.8.2 - b3092 -————————————— -- Fix: Davis VP2 stations - LastRainTip not being updated correctly with mixed metric/imperial gauges/units -- Fix: Davis VP2 stations - Bug that prevented the seconds being read from the console clock -- Fix: Dashboard temp/press trend arrows not working in locales with comma decimals -- MXdiags log files are now rotated if they exceed 20MB in size, 15 log files are now retained -- For Davis VP2 stations, setting the console clock now checks the console time and only sets it if it - differs from the current time by greater than 60 seconds -- Davis VP2 stations protocols have been extensively rewritten to make them more efficient and less timing sensitive - Third party IP loggers such as meteobridge and WiFi logger should now work more reliably -- Enables PSK authentication when using SFTP. Options are now Password, PSK, PSK with Password fall-back - New Cumulus.ini entries... - [FTP site] - SshFtpAuthentication=password [password (default), psk, password_psk] - SshFtpPskFile= [PSK filename, must include the full path is the file is not in the CumulusMX folder] -- CMX calculated dew point is automatically enabled for GW1000 stations - -- Updated files - \CumulusMX.exe - \interface\js\dashboard.js - \interface\json\InternetOptions.json - \interface\json\InternetSchema.json - - -3.8.1 - b3091 -————————————— -- Fix: Davis TCP loggers failing to connect correctly -- Fix: NOAA reports not viewable in Admin interface when Cumulus started as a service -- Fix: Prevents an Exception on shutdown when running as a service - -- Updated files - \CumulusMX.exe - - -3.8.0 - b3090 -————————————— -- Fix: Malformed API JSON returned by api/tags/process.json if no data output is generated -- Fix: Crash on start-up on the 1st of the month between 00:00 and rollover hour when using 9am rollover -- Fix: AWEKAS Sunshine hours were being truncated to integer hours -- New feature: Cumulus MX now supports running as a system service - - When running as a service the normal console output is logged to a file: \MXdiags\ServiceConsoleLog.txt - - On Windows - - To set Cumulus up as a service run it as an Administrator using the command line parameter -install - - To remove the Cumulus system service run it as an Administrator using the command line parameter -uninstall - - The service will stop itself if it detects that the computer is going into standby - - To add parameters to the service when it starts automatically you will have to edit the registry - HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\CumulusMX\ImagePath - Add the parameters after the quoted executable string. Eg. - "C:\CumulusMX\CumulusMX.exe" -port 9000 - - The service can be stopped/started using the usual utilities: - The Services applet:- services.msc (You can pass parameters using this tool, but they are not saved for future use) - The service command line tool:- sc start|stop|query CumulusMX [optional parameters for start-up here] - - On Linux - - Run Cumulus using the mono-service command instead of mono. You also need to add the command line parameter -service - > mono-service -l:/var/run/cmx.pid CumulusMX.exe -service (if pathed to CumulusMX folder) - or - > mono-service -l:/var/run/cmx.pid -d:<> CumulusMX.exe -service (from any location) - - To stop the service you use kill, which will perform an orderly shutdown of MX - > kill `cat /var/run/cmx.pid` -- New feature: - - When running on Linux there is now an init.d script to start/stop the service. - - The script can be found in the \MXutils\linux folder - - Copy it to your /etc/init.d folder - - Change the permissions to 0755 - > chmod 0755 /etc/init.d/cumulusmx.sh - - The script supports the following commands: start, stop, restart, status, removelock - - start: Starts MX as a service and creates a lock file to prevent it running again - - stop: Stops MX as a service and removes the lock file - - restart: Stops, then starts MX - - status: Reports if MX is running (only by checking the lock file at present), and dumps the latest console output to screen - - removelock: Removes a stray lock file if one gets left behind - - The command line would be: /etc/init.d/cumulusmx.sh start|stop|restart|status|removelock - - Script name: \MXutils\linux\cumulusmx.sh -- New feature: - - When running on Windows as a service, the service will stop if the computer goes to standby/hibernate mode. A script to set up - a Scheduled Task that will restart the service on resume from standby/hibernate is provided in the \MXutils\windows folder. - The script MUST be run with Administrator privileges. - - Script name: \MXutils\windows\CreateCmxResumeFromStandbyTask.ps1 -- Adds a new web tags for the Davis WLL device. - <#ConsoleSupplyV> - The WLL external supply voltage - <#MulticastBadCnt> - Count of multicast packets missed or have bad a CRC - <#MulticastGoodCnt> - Count of good multicast packets received - <#MulticastGoodPct> - Percentage of good multicast packets received -- On new installs of Cumulus MX, changes the default value of "WarnMultiple" from false to true -- Adds a download history limit of 50 bad packets for WMR200 station types. Previously it could get stuck in a loop downloading bad data. -- Adds Show UV-I to the graph hide/show options -- Updates ExportMySQL to include the new Humidex values from b3089 -- AWEKAS uploads updated to API v4 - now can include indoor data, soil temp/moisture 1-4, and leaf wetness 1-2. And it supports rapid - updates - minimum 15 seconds if you have a Stationsweb subscription, min 60 secs for Pro accounts, and 300 seconds for normal accounts - - IMPORTANT NOTE: THe AWEKAS interval has changed from being set in minutes, to being set in seconds. - You MUST update the AWEKAS interval in the Internet settings/Cumulus.ini file to reflect this change. - - New Cumulus.ini entries - [Awekas] - Language=en - SendIndoor=0 - SendSoilMoisture=0 - SendLeafWetness=0 -- Enabling/disabling and changing intervals for Twitter, WUnderground, Windy, PWS Weather, WOW, APRS, AWEKAS, Weather Cloud, MQTT from - Internet Settings now takes immediate effect and does not require a Cumulus restart. -- Adds a new Cumulus.ini file only, read-only setting - [Station] - SunshineHrsDecimals=1 -- Adds a new folder in Dist zip file: /MXutils - In here I will add the various scripts etc that are provided with the releases to update MySQL or aid moving from one release to the next. - See above for new Windows and Linux scripts when running as a service - - -- Updated files - \CumulusMX.exe - \ExportMySQL.exe - \interface\js\charts.js - \interface\json\InternetOptions.json - \interface\json\InternetSchema.json - \interface\json\StationOptions.json - \interface\json\StationSchema.json - \webfiles\js\cumuluscharts.js - --New files - \MXutils - - -3.7.0 - b3089 -————————————— -- Fix: Humidex was treated by MX as a temperature and incorrectly converted using the Celsius to Fahrenheit - conversion (if that was the user selected temperature unit). - Humidex is a dimensionless scale that should not be converted to retain meaning -- Fix: Thread unsafe graph data access -- Fix: Real time SFTP not recovering from some reconnection situations -- Fix: Add missing "rc" function from <#battery> web tag -- Fix: Remove units from Humidex value on the chart tooltip -- Fix: Station Settings page menu -- Fix: Davis WLL devices were flagging sensor contact lost for lost packets, now it only flags contact lost - when resynchronising with a transmitter -- Fix: Monthly records editor not functioning correctly on Humidex in beta1 -- Adds a logger interval check for Davis VP2 stations - prints warning to console and log file if there is a mismatch -- Now supports some of the Cumulus 1 Graph plotting options, configured via Station Settings - - Cumulus.ini file settings - [Graphs] - TempVisible=1 - InTempVisible=1 *new? - HIVisible=1 - DPVisible=1 - WCVisible=1 - AppTempVisible=1 - FeelsLikeVisible=1 - HumidexVisible=1 *new - InHumVisible=1 - OutHumVisible=1 -- Remove defunct WeatherBug upload from Cumulus MX -- Humidex is now a fully fledged variable... - - Daily, monthly, yearly highs - - Recent data - - Graph data - added to MX interface - - Records - - New Humidex web tags: - <#HighHumidexRecordSet> - <#humidexTH>, <#ThumidexTH> - <#humidexYH>, <#ThumidexYH> - <#MonthHumidexH>, <#MonthHumidexHT>, <#MonthHumidexHD> - <#YearHumidexH>, <#YearHumidexHT>, <#YearHumidexHD> - <#humidexH>, <#ThumidexH> - <#ByMonthHumidexH>, <#ByMonthHumidexHT> - <#RecentHumidex> -- Add Humidex to default web site trend graphs -- Now only calculates Humidex above 10C -- Day file updated with four additional fields or Humidex high, high time -- Monthly log file has an additional field for Humidex -- Daily SQL table has four additional columns for Humidex high, high time -- Monthly SQL table has an additional column for Humidex -- Add new web tag <#timeUnix> = provides current date/time as a Unix timestamp (like <#timeJavaScript>) -- The web tags <#SunshineHours> and <#YSunshineHours> now accept the decimal place parameter "dp=n" -- Adds a new web token processor API - details available separately -- Spike removal settings are now active for all station types - TAKE CARE! Use with caution -- Spike/limit logging is now enabled by default for new installs -- Adds a new alarm - Spike removal triggered - it stays active for 12 hours, or it is cleared if you save the calibration settings - - New web tag <#DataSpikeAlarm> - - New Cumulus.ini file settings... - [Alarms] - DataSpikeAlarmSet=0 - DataSpikeAlarmSound=0 - DataSpikeAlarmSoundFile=alarm.mp3 -- New "Limits" implemented. Now MX ignores temperature, dew point, pressure, and wind values outside sensible value ranges from any station type (these also trigger a spike alarm) - - Set via the Calibration settings screen - - New Cumulus.ini file settings... - [Limits] - TempHighC=60 - TempLowC=-60 - DewHighC=40 - PressHighMB=1090 - PressLowMB=870 - WindHighMS=90 -- New web tags for latest build notification - these require internet access to be meaningful - <#NewBuildAvailable> - returns "1" or "0" - <#NewBuildNumber> - returns the latest available Cumulus MX build number - - -- Updated files - \CumulusMX.exe - \dayfileheader.txt - \monthlyfileheader.txt - \interface\alarmsettings.html - \interface\alltimerecseditor.html - \interface\index.html - \interface\monthlyrecseditor.html - \interface\stationsettings.html - \interface\thismonthrecseditor.html - \interface\thisyearrecseditor.html - \interface\js\alarmsettings.js - \interface\js\alltimerecseditor.js - \interface\js\charts.js - \interface\js\dashboard.js - \interface\js\datalogs.js - \interface\js\dayfileeditor.js - \interface\js\monthlyrecseditor.js - \interface\js\thismonthrecseditor.js - \interface\js\thisyearrecseditor.js - \interface\json\CalibrationOptions.json - \interface\json\CalibrationSchema.json - \interface\json\InternetOptions.json - \interface\json\InternetSchema.json - \webfiles\js\cumuluscharts.js - - -3.6.12 - b3088 -—————————————— -- Fix Davis stations not downloading historic logger data (in b3087) - -- Updated files - \CumulusMX.exe - - -3.6.11 - b3087 -—————————————— -- Fix Davis TCP logger connections not timing out occasionally -- Fix heading on interface Now page, remove units from Humidex -- Fix FTP log file handling in Extra Files, with EOD option on the first of the month -- Add Feels Like to the default web site trends temperature graph -- Add Extra Sensors log file to the backup routine -- Add previous months log files (monthly and extra) to the backup on the first of the month -- Add "" tag to Extra Web Files to specify the variable extra log file name -- Improve web tag token parser performance -- Cumulus (Zambretti) forecast now works with localised compass points -- Internal optimisations (watch out for new issues!) -- Uplift the SFTP component from a 2016 version to new beta version - supports more encryption methods and key file formats -- Further additions to shutdown code for all stations -- Adds new web tag <#RecentFeelsLike> - -- Updated files - \CumulusMX.exe - \Renchi.SshNet.dll - \interface\now.html - \webfiles\js\cumuluscharts.js - - -3.6.10 - b3086 -—————————————— -- Fix for Feels Like calculation broken in previous release -- Fix for Davis WLL wind values when using units other than mph -- Fix for poor performance of wind direction charts on the MX interface and base web site -- Make end of day SQL inserts asynchronous -- Use a fixed timestamp for all EOD operations - -- Updated files - \CumulusMX.exe - \interface\charts.html - \interface\js\charts.js - \web\trendsT.htm - \webfiles\js\cumuluscharts.js - -3.6.9 - b3085 -————————————— -- RELEASE WITHDRAWN - - -3.6.8 - b3084 -————————————— -- Simplify realtime SFTP error detection and recovery -- Change the default web site Gauges page to not show pop-up graphs by default -- Fix for Ecowitt GW1000 stations when sensors go offline/online (wind and rain values) -- Fix for GW1000 stations wind gust values when using units other than "mph" -- Fix for GW1000 stations with WH34 type sensors and firmware 1.6.0 or later. You *must* now use firmware 1.6.0+ with WH34 devices -- Fix crash when creating the graph JSON files when file in use by FTP -- Fix for rc=y parameter not working with the <#intemp> web tag -- Fix low contrast menus on admin interface -- Fix HighCharts theme on admin interface Charts page, and default web site Trends page -- Fix for web tags <#daylength> and <#daylightlength> to display "24:00" if they last all day (they still allow custom formats) - -- Updated files - \CumulusmX.exe - \interface\charts.html - \interface\css\cumulus.css - \web\trendsT.htm - \webfiles\lib\steelseries\scripts\gauges.js - - -3.6.7 - b3083 -————————————— -- Add catches for real time MySQL updates and all real time file failures -- Adds Station (Absolute) and Altimeter pressure values for Davis WLL stations - -- Updated files - \CumulusMX.exe - - -3.6.6 - b3082 -————————————— -- Change ini files to use 17 significant figures for decimal values (up from 15) -- Fix for Davis WLL health data decoding when the WLL is LAN attached -- Fix for real time SFTP not reconnecting after failure - -- Updated files - \CumulusMX.exe - - -3.6.5 - b3081 -————————————— -- Fix for sun rise/set and dawn/dusk calculations when there is one event but not the other in a single day -- Fix for realtime FTP timeout/recovery issues - -- Updated files - \CumulusMX.exe - - -3.6.4 - b3080 -————————————— -- Fix for Ctrl-C not being handled when running under Linux/mono. Now handles SIGTERM and console Ctrl-C -- Fix for realtime FTP getting stuck on "already in progress" -- Adds support for Ecowitt GW1000 WH34 8 channel "User" (soil and water) temperature sensors - New web tags <#UserTemp1> - <#UserTemp8> - ExtraLog file has eight new fields appended - UserTemp1-8 - -- Updated files - \CumulusMX.exe - \Extrafileheader.txt - \SampleStrings.ini - \interface\extra.html - \interface\js\extradatalogs.js - \interface\js\extrasensors.js - - -3.6.3 - b3079 -————————————— -- Reverts b3077 FluentFTP update -- Fix for the long standing random Cumulus.ini/today.ini corruption when shutting down on Windows -- Another fix for Oregon WMR928 extra temperature only sensors - - -- Updated files - \CumulusMX.exe - \FluentFTP.dll - - -3.6.2 - b3078 -————————————— -- Fix for badly formed realtime.txt in b3077 - - -- Updated files - \CumulusMX.exe - - - -3.6.1 - b3077 -————————————— -- Fix for Oregon WMR928 extra temperature only sensors -- Fix for yesterdays Feels Like values in Admin interface Today/Yesterday screen -- Adds Feels Like to realtime.txt file as field 59 -- Changes GW1000 default Lightning distance to 999 (all user units), and time to 1900-01-01 00:00:00 - The corresponding Webtags will output "--" and "---" respectively -- Adds a new web tag <#LastRainTip>, which unlike <#LastRainTipISO> will accept a date/time format string - - -- Updated files - \CumulusMX.exe - \FluentFTP.dll - - - -3.6.0 - b3076 -————————————— -- Fix for records editors failing to read log files from Cumulus 1 versions -- Fix for Ecowitt GW-1000 devices to bypass the auto-discovery mechanism if it is disabled in the config -- NOAA reports now include degree, minutes, seconds symbols -- Slightly enhanced program termination logging -- Implements highs/low/records for Feels like - - Changes to ini files to add Feels Like - Adds [FeelsLike] section to - today.ini, yesterday.ini, month.ini, year.ini, alltime.ini, monthlyalltime.ini - - Changes to log files - monthlog.txt - adds field 27, feels like - dayfile.txt - adds fields 47-50, high feels like, high time, low, low time - - Changes to the MYSQL database tables are required - Adds column FeelsLike to the Monthly table - Adds columns MaxFeelsLike, TMaxFeelsLike, MinFeelsLike, TMinFeelsLike to the Dayfile table - - New web tags - <#HighFeelsLikeRecordSet> - <#LowFeelsLikeRecordSet> - <#ByMonthFeelsLikeHT> - <#ByMonthFeelsLikeLT> - <#ByMonthFeelsLikeL> - <#ByMonthFeelsLikeH> - <#YearFeelsLikeHD> - <#YearFeelsLikeLD> - <#YearFeelsLikeHT> - <#YearFeelsLikeLT> - <#YearFeelsLikeH> - <#YearFeelsLikeL> - <#MonthFeelsLikeHD> - <#MonthFeelsLikeLD> - <#MonthFeelsLikeHT> - <#MonthFeelsLikeLT> - <#MonthFeelsLikeH> - <#MonthFeelsLikeL> - <#feelslikeH> - <#TfeelslikeH> - <#feelslikeL> - <#TfeelslikeL> - <#feelslikeYH> - <#TfeelslikeYH> - <#feelslikeYL> - <#TfeelslikeYL> - <#feelslikeTH> - <#TfeelslikeTH> - <#feelslikeTL> - <#TfeelslikeTL> - - Updated record editors - - Updated log file viewers/editors -- Adds battery and reception data for Davis WLL. It now logs battery and input voltages to the MXdiags. - These are updated every 15 minutes and require you to have a WeatherLink Pro subscription. - The WLL unlike the VP2 console provides individual data for each transmitter - - The following web tags have been updated to accept a "tx=n" parameter, where n=1-8 and equals the desired transmitter id. - Omitting the tx= parameter or using tx=0 makes the tag function as before for Davis VP2 systems - <#DavisTotalPacketsMissed tx=n> - <#DavisNumberOfResynchs tx=n> - <#DavisMaxInARow tx=n> - <#DavisNumCRCerrors tx=n> - - New web tags for WLL transmitter reception percentage and RSSI figure, these must be used with the tx=n parameter - <#DavisReceptionPercent tx=n> - defaults to tx=1, tx=0 is unused - <#DavisTxRssi tx=n> - defaults to tx=1, use tx=0 to get the WLL WiFi RSSI -- Updated ExportMySQL.exe to version 1.1.0 - - Incorporates the new Feels Like data - - Uses compass point "-" for Calm - - Reads customised compass points from strings.ini if set - - -- Updated files - \CumulusMX.exe - \dayfileheader.txt - \ExportMySQL.exe - \Extrafileheader.txt - \monthlyfileheader.txt - \interface\alltimerecseditor.html - \interface\extrawebfiles.html - \interface\index.html - \interface\monthlyrecseditor.html - \interface\now.html - \interface\thismonthrecseditor.html - \interface\thisyearrecseditor.html - \interface\todayest.html - \interface\js\alltimerecseditor.js - \interface\js\charts.js - \interface\js\datalogs.js - \interface\js\dayfileeditor.js - \interface\js\monthlyrecseditor.js - \interface\js\thismonthrecseditor.js - \interface\js\thisyearrecseditor.js - \interface\json\StationOptions.json - - - -3.5.4 - b3075 -————————————— -- Fix for bearing zero on the interface "Now" page -- Fix for admin interface charts popup -- Fix for "normal" Extra files not being FTP'd at rollover interval, only those flagged as EOD were being transferred -- Another attempt to rationalise the Moon Phase messages - each quarter (new, 1st, full, 3rd) should now show for approximately 12 hours either side of the event -- All web tags that produce decimal number output now support the "rc=y" option -- Additional Davis WLL health info dumped in the MXdiags log on start-up and every 10 minutes if debug logging is on (voltages, uptimes, WiFi RSSI etc) - Logs warnings to the command line console and log file if voltages are too low -- Adds BatteryLow Alarm for WLL (Console [if you have an API key] & Tx), VP2 (Console & Tx), GW1000 (Tx), and adds a new web tag <#BatteryLowAlarm> -- Adds new web tag <#feelslike> - calculated using the JAG/TI formula used in the UK, USA, Canada etc. Currently there are no stats for this value - -- Updated files - \CumulusMX.exe - \interface\alarmsettings.html - \interface\alltimerecseditor.html - \interface\index.html - \interface\now.html - \interface\todayest.html - \interface\js\alarmsettings.js - \interface\js\charts.js - \interface\js\dashboard.js - \interface\js\now.js - \mqtt\DataUpdateTemplate.txt - \mqtt\IntervalUpdate.txt - \webfiles\js\cumuluscharts.js - - - -3.5.3 - b3074 -————————————— -- Fix: Backs out changes that created bad file paths in b3073 - - -- Updated files - \CumulusMX.exe - \web\webfiles\lib\steelseries\scripts\gauges.js - - - -3.5.2 - b3073 -————————————— -- Fixes and improvements to MQTT processing -- Adds new Cumulus.ini setting for MQTT to force IPv4/IPv6 connectivity (default is System decides which to use) - [MQTT] - IPversion=0 (0=default, 4=IPv4, 6=IPv6) -- Adds Sensor Contact lost flag/alarm for Davis WLL devices -- GW1000 raw Lux value is now available via the <#Light> web tag like other Fine Offset stations -- Adds three new web tags... - <#timeJavaScript> - returns the current date/time in JavaScript milliseconds. Example use = "var dt = Date(<#timeJavaScript>)" - <#directionTM> - returns todays max wind gust direction as a compass point - <#directionYM> - returns yesterdays max wind gust direction as a compass point - - -- Updated files - \CumulusMX.exe - \MQTTnet.dll - - - -3.5.1 - b3072 -————————————— -- Fix for the "Stop second instance" option now working a bit too well, you could not disable it! -- Implements the "record set" web tags. These will be set from the time of the record until a timeout value (default 24 hours). - You can change the default timeout by adding a entry to Cumulus.ini - [Station] - RecordSetTimeoutHrs=24 - - The web tags enabled are: TempRecordSet, WindRecordSet, RainRecordSet, HumidityRecordSet, PressureRecordSet, HighTempRecordSet, - LowTempRecordSet, HighAppTempRecordSet, LowAppTempRecordSet, HighHeatIndexRecordSet, LowWindChillRecordSet, HighMinTempRecordSet, - LowMaxTempRecordSet, HighDewPointRecordSet, LowDewPointRecordSet, HighWindGustRecordSet, HighWindSpeedRecordSet, HighRainRateRecordSet, - HighHourlyRainRecordSet, HighDailyRainRecordSet, HighMonthlyRainRecordSet, HighHumidityRecordSet, HighWindrunRecordSet, LowHumidityRecordSet, - HighPressureRecordSet, LowPressureRecordSet, LongestDryPeriodRecordSet, LongestWetPeriodRecordSet, HighTempRangeRecordSet, LowTempRangeRecordSet - - -- Updated files - \CumulusMX.exe - - - -3.5.0 - b3071 -————————————— -- Fix to "Stop second instance" of Cumulus running -- Fix for hung update interval (S)FTP sessions getting hung -- Fix to <#moonage> web tag to improve accuracy -- Adds support for MQTT output - Two options, send an MQTT message whenever new data is received, or send a message at a fixed interval - The message format is defined in template files, using web tags, located in the \mqtt folder -- Tidy up of /interface folder to remove unused files -- Removal of Highcharts scripts from the distribution -- Add support for generating a Moon Phase image. This is disabled by default. The output image will be generated and - optionally FTP'd once an hour. The generated local image file is always \web\moon.png - [FTP site] - IncludeMoonImage=1 - - [Graphs] - MoonImageEnabled=1 - MoonImageSize=100 - MoonImageFtpDest=images/moon.png - - -- New files - \Licences-Additional.txt - \MQTTnet.dll - \mqtt\DataUpdateTemplate.txt - \mqtt\IntervalTemplate.txt - \web\MoonBaseImage.png - -- Updated files - \CumulusMX.exe - \FluentFTP.dll - \Newtonsoft.Json.dll - \interface\ - [Many changes, delete and replace] - \web\indexT.htm - \webfiles\js\cumuluscharts.js - - - -3.4.6 - b3070 -————————————— -- Fix for station wind chill on Davis WLL devices -- Fix for auto-discovered Davis WLL Station-Ids not being saved to the config file -- More robust failure handling for realtime FTP connections -- Additional diagnostics output in the console and log file for badly formed web tags - -- Updated files - \CumulusMX.exe - - - -3.4.5 - b3069 -————————————— -- Adds Editors for: Dayfile, Monthly Logs, Extra Logs -- Adds line numbers to the log file viewer/editors -- Widens the time windows for the Moons phase names -- Fix for <#MoonPercent> and <#MoonPercentAbs> always showing integer values even with the dp=n option - - -- Updated files - \CumulusMX.exe - \interface\ - - - -3.4.4 - b3068 -————————————— -- Fix for incorrect NOAA yearly report, annual averages for temperature and wind were calculated incorrectly -- Now detects invalid CumulusMX.exe command line parameters -- Adds a new command line parameter -debug. This switches on debug and data logging from the start-up of Cumulus MX. You no - longer have to edit Cumulus.ini to gather these diagnostics. - -- Updated files - \CumulusMX.exe - - - -3.4.3 - b3067 -————————————— -- Adds a new option to Davis WLL and Ecowitt GW1000 station settings to disable IP address auto-discovery. Use this option - if you have more than one of these devices on your network, then enter the IP address manually. - -- Updated files - \CumulusMX.exe - \interface\json\StationOptions.json - \interface\json\StationSchema.json - - - -3.4.2 - b3066 -————————————— -- Improved error handing for invalid Davis WLL Station Ids -- Improved error handling when the network connection to a Davis WLL is lost (and restored) -- Adds missing Data Stopped alarm to the Dashboard and Alarm Settings screens -- Adds auto-discovery for Ecowitt GW1000 devices IP addresses -- Adds DataStopped handling to Ecowitt GW1000 devices - -- Updated files - \CumulusMX.exe - \interface\alarmsettings.html - \interface\index.html - \interface\js\alarmsettings.js - \interface\js\dashboard.js - - - -3.4.1 - b3065 -————————————— -- Fix for WLL if you change the WL.com logging interval around a catch-up period -- Fix for gust values from WLL devices -- Add WLL broadcast data watchdog and warning, implements DataStopped flag -- Adds WLL Cumulus.ini readonly setting, AutoUpdateIpAddress, use this to switch off the WLL autodetection of IP - address. This is a workaround for a WLL firmware bug that does not update the IP address when it changes using DHCP. - AutoUpdateIpAddress=0 #default = 1 - -- Updated files - \CumulusMX.exe - - - -3.4.0 - b3064 -————————————— -- Adds the option for Davis WLL users who have a WL.com Pro subscription to use WL.com as a "logger" to catch up - missing data on Cumulus MX start-up. -- Adds to option to truncate the <#MoonAge> tag value to an integer value instead of rounding it. - Use <#MoonAge tc=y> - ideal if you use the tag for image selection -- Updates FTP Now so that it does a full file process and FTP cycle, previously it just ran the FTP process -- Adds a Cumulus MX version check at startup - if online - and logs a message in the console and diags when a newer - build is available -- Fixes the Monthly Records editor for dry/wet periods that end on the last day of a month being incorrectly recorded - against the following month. -- Fixes the Beaufort calculations - there were some rounding errors in edge cases. -- Fixes Davis VP2 and WLL that were not using peak speeds from LOOP2 (VP2) and live/historic data (WLL) when the Cumulus logging - interval is set to 10 minutes or more. -- Fix for FTPS on realtime FTP updates -- Adds new ini file only option to disable Explicit FTPS - ie use Implicit mode - DisableFtpsExplicit=1 - -- Updated files - \CumulusMX.exe - \interface\json\StationOptions.json - \interface\json\StationSchema.json - - -3.3.0 - b3063 -————————————— -- Adds support for SFTP (SSH FTP) - - Moves the FTP SSL option to the web server settings - - Adds SFTP to existing FTP, FTPS options -- Fixes Ecowitt Soil Temperature/Moisture/Leak detector channel numbering -- Fix for an occasional error in station logger data handling of today's rainfall during CMX start-up -- Fix for Ecowitt GW1000 Lightning data decode -- Fix for incorrect date on <#Snow*> web tags -- Now automatically fixes two Cumulus.ini changes from Cumulus 1 generated files - - Changes the [FTP Site] section name to [FTP site] - - Changes NOAA default monthly name (if still set) from "NOAAMO'mmyy'.txt" to "NOAAMO'MMyy'.txt" -- Additional diagnostic logging info for Lacrosse WS2300 stations - -- Updated files - \CumulusMX.exe - \FluentFTP.dll - \interface\json\InternetOptions.json - \interface\json\InternetSchema.json - -- New files - \Renci.SshNet.dll - - -3.2.6 - b3062 -————————————— -- Fixes monthly records editor for stations with a met day starting at 9am -- Adds range checks for latitude and longitude values - -- Updated files - \CumulusMX.exe - - -3.2.5 - b3061 -————————————— -- Adds This Month and This Year records editors -- Adds FTP Now function -- Fix to MonthlyAlltimeLog.txt to add line feeds -- Fix missing WLL station description from APRS data - -- Updated files - \CumulusMX.exe - \interface\.html - -- New files - \interface\ftpnow.html - \interface\thismonthrecseditor.html - \interface\thisyearrecseditor.html - \interface\js\thismonthrecseditor.js - \interface\js\thisyearrecseditor.js - - -3.2.4 - b3060 -————————————— -- Fix uncaught Web exceptions in Davis WLL -- Fix Monthly Records editor not saving updated date/time stamps -- Adds a log file for MonthlyAlltime.ini file changes - -- Updated files - \CumulusMX.exe - - -3.2.3 - b3059 -————————————— -- Adds the Cumulus.ini file to the files automatically backed up each day/program start -- Fixes to the Monthly Records editor monthly rainfall figures - -- Updated files - \CumulusMX.exe - \interface\alltimerecseditor.html - \interface\monthlyrecseditor.html - \interface\js\alltimerecseditor.js - \interface\js\monthlyrecseditor.js - - -3.2.2 - b3058 -————————————— -- Implements the missing <#txbattery> web tag for WLL devices -- Fix default website pages header not wrapping on small screens -- Adds Monthly Records editor -- Fixes and improvements to the All Time Records editor - -- Updated files - \CumulusMX.exe - \interface\.html - \web\T.htm - -- New files - \interface\monthlyrecseditor.html - \interface\js\monthlyrecseditor.js - - -3.2.1 - b3057 -————————————— -- Fix for WMR200 stations writing a zero value Apparent Temperature to the log files when retrieving logger data -- Fix the dashboard for Internet Explorer -- Fix default website index page header not wrapping on small screens -- Fix for Davis stations connected via TCP/IP to detect failures and reopen the connection more quickly during loop data processing -- Adds Solar calibration settings offset and multiplier -- Updates the SampleStrings.ini file with the extra captions added in b3056 - -- Updated files - \CumulusMX.exe - \SampleStrings.ini - \HidSharp.dll - \interface\js\dashboard.js - \interface\json\CalibrationSchema.json - \web\indexT.htm - - -3.2.0 - b3056 -————————————— -- Adds support for Ecowitt GW1000 WiFi gateway -- New web tag <#GW1000FirmwareVersion> -- Added extra soil temp sensors 5-16 for GW1000 stations - - New web tags <#SoilTemp5-16> - - Value are logged to extra log file - - Added custom captions available in Cumulus.ini [SoilTempCaptions] : Sensor5-16 -- Added extra soil moisture sensors 5-16 for GW1000 stations - - New web tags <#SoilMoisture5-16> - - Values are logged to extra log file - - Added custom captions available in Cumulus.ini [SoilMoistureCaptions] : Sensors5-16 - - Units now change between cb for Davis and % for GW1000 sensors -- Added extra air quality sensors 1-4 for GW1000 stations - - New web tags <#AirQuality1-4> and <#AirQualityAvg1-4> - - Values are logged to extra log file - - Added custom captions available in Cumulus.ini [AirQualityCaptions] : Sensor1-4 and SensorAvg1-4 -- Added new web tags for <#LeakSensor1-4> to support GW1000 leak sensors - - No display or logging of these values is done -- Added new web tags for Lightning sensors on GW1000 stations - - <#LightningDistance> Distance to last strike (same units as wind run - miles/km/nm) - - <#LightningTime> Time of last strike (format customisable) - - <#LightningStrikesToday> Number of strikes since midnight - - Currently there is no logging or display of these values -- Enables alarms as per Cumulus 1 - - New Alarm page under Settings - - Alarms are shown visually on the dashboard - - Due to browser restrictions, alarm sounds on the browser page may require you to click a button on the first alarm in order to hear it. - - You can add the MX admin site to your browsers list of sites allowed to play sound automatically - - Your browser should "learn" that you want to allow sounds to play automatically - - Alarm sound files should be placed in the /interface/sounds folder, they must be a browser compatible format (mp3 are good) - - The alarm settings for the sound file should be just the filename without any path -- Add pressure multiplier calibration - - New ini file setting [Offsets] PressMult = 1.0 (default) -- Fix for the All Time Records editor hourly rain total from the monthly log file month -- Fixes missing Cumulus multipliers for Wind and Rain values on Davis WLL stations -- Changes All Time Records editor to only load data from the Day File by default. Monthly log file - processing is now optional as it can take a very long time on a slow machine e.g. Raspberry Pi -- Fix to catch badly formed Davis WLL broadcast messages -- Fix for Davis WLL edge cases producing an initial zero value wind chill on startup - -- Updated files - \CumulusMX.exe - \interface\json\StationOptions.json - \interface\json\StationSchema.json - \interface\.html - \interface\js\extrasensors.js - \interface\js\alltimerecseditor.js - \interface\js\dashboard.js - \interface\css\main.css - \interface\json\CalibrationSchema.json - -- New files - \interface\alarmsettings.html - \interface\js\alarmsettings.js - \interface\sounds\alarm.mp3 - - -3.1.2 - b3055 -————————————— -- Fix for the All Time Records editor monthly rain total from the monthly log file month -- Fix for some long timeouts in All Time Records editor - -- Updated files - \CumulusMX.exe - - - -3.1.1 - b3054 -————————————— -- Fixes WLL timestamps always being in UTC, now uses local time -- Adds web tags <#snowlying>, <#snowfalling>, both provide 1|0 responses -- Adds Current Conditions editor to admin interface -- Adds All Time Records editor to admin interface - - -- Updated files - \CumulusMX.exe - \interface\.html - --New files - \interface\currentcondeditor.html - \interface\alltimerecseditor.html - \interface\js\alltimerecseditor.js - \interface\lib\x-editable\ - \interface\img\loading.gif - - -3.1.0 - b3053 -————————————— -- Adds support for Davis WeatherLink Live device - - Supports Zero-Config, it should discover your WLL on the network (router support required) -- Adds support for Cumulus.ini file setting EWpressureoffset for Fine Offset stations as used in Cumulus1. - - This provides a manual override for the calculated absolute to relative pressure offset -- Adds reading of the interval Hi/Lo temperature readings when processing Davis logger archive records during catch-up -- Adds display and generation of NOAA Monthly and Yearly reports -- Reduces the Instromet live data read intervals to 1 second -- Applies "fix" for Mono 5.x generating short month names ending with "." as used for log file names - - -- Updated files - \CumulusMX.exe - \CumulusMX.exe.config - \interface\.html - \interface\json\StationOptions.json - \interface\json\StationSchema.json - -- New files - \Tmds.MDns.dll - - -3.0.2 - b3052 -————————————— -- Fixes Davis archive downloads from the the logger when the day rollover processing takes longer than 10 seconds. - This can happen on slow processors - Pi Zero for example - or if lengthy procedures are included - An extra archive processing run is scheduled for each day rollover that takes longer than 10 seconds - -- Updated files - \CumulusMX.exe - \CumulusMX.exe.config - - -3.0.1 - b3051 -————————————— -- Increases Davis DMPAFT Date/Time command timeout -- Much improved Davis serial port throughput (for USB and Serial loggers), should decrease the historic logger download time -- Adds Davis archive data processing progress indication -- Fixes firmware check for LOOP2 support -- Fixes a race condition on start-up of Davis VP2 stations without a logger. This could cause a crash in AstroLib.SolarMax() -- Fixes reading the Davis console clock after setting the time - -- Updated files - \CumulusMX.exe - - -3050 -———— -- Fixes MX not working with locales that use two character date separators - Eg. Croatia "29. 04. 19" - -- Updated files - \CumulusMX.exe - -3049 -———— -- Adds the ability to upload data to Windy.com - -- Updated files - \CumulusMX.exe - \interface\json\InternetOptions.json - \interface\json\InternetSchema.json - -3048 -———— -- You can now first time enable/disable Realtime FTP without having to restart CMX - -- Instromet stations now record and report rainfall (mm) and sunshine hours to 2 decimal places - -- Improved realtime FTP error handling - -- Improved Davis protocol handling - -- Fix Davis protocol mixing up LOOP1 and LOOP2 packets and consequently providing invalid rain and wind data. - -- Fix web tag <#YearLowDailyTempRangeD> broken in b3047 - -- Bug fixes to FTP Component, and internal changes to FTP transfer mechanism - -- Updated files - \CumulusMX.exe - \FluentFTP.dll - - - -3047 -———— -- Web token parser updated to cope with html tag characters "<>" in the format string. - - You can now do things like... - <#TapptempH format="dd' 'MMM' 'yyyy' at 'HH:mm''"> - which gives... - 04 Dec 2018 at 10:12 - - Note: that you have to use single quotes for HTML entity names, and they have to be escaped "\'" - -- New Davis Baud Rate setting - - Allows you to alter the speed of the serial connection to Davis loggers - - Configured manually in Cumulus.ini [Station] section - DavisBaudRate=19200 (default) Permitted values are 1200, 2400, 4800, 9600, 14400, 19200 - -- Added new option for the "Extra files" - End of Day - - Enabling this means that file will only be processed/copied/FTPed once a day during the end of day roll-over. - - There is a new Cumulus.ini file setting for each "extra" file associated with this setting - ExtraEOD[nn] - - Note there is currently no check between Realtime and End of Day settings, you could check both options and the file - will be processed at both the realtime interval AND end of day - which would not make much sense! - -- Improvement to Instromet logger protocol handling - -- Change the Fine Offset Synchronised Reads option to default to enabled - -- Change VP2 automatic disabling of LOOP2 to an advisory message, as the firmware version is not always detected. - -- Consistency: All record Value tags should now return '---' and Date tags '----' until they are first set. - -- The following web tags now support the "dp=N" "rc=y" parameters for the number of decimal places, and replace decimal commas with points. - #MoonPercent - #MoonPercentAbs - #MoonAge - -- Fix for Fine Offset & WMR100/200 stations on Mac operating systems (introduced in b3044) - -- Fix for invalid (extremely high) pressure readings from Fine Offset stations (thanks to 79669weather) - -- Fix to not updating the Instromet loggers memory pointer correctly - -- Fixed Weather Diary Time Zone issues - -- Bug fixes and performance improvements to the FTP component - -- Updated files - \CumulusMX.exe - \CumulusMX.exe.config - \FluentFTP.dll - \HidSharp.dll - \interface\js\diaryeditor.js - \interface\json\StationOptions.json - -- Removed files - \fastJSON.dll - - - -3046 -———— -- Weather Diary - - Added Weather Diary page to management interface - - Added diary.db file to daily backup files - - Removed diary.db from distribution (it is created on first use to avoid overwriting the file on CMX distro updates) - -- Web Tags - - Added <#snowdepth> tag processing - -- ET annual roll-over fix - -- Fix to TLS 1.2 FTPS of the 'periodic' files - - - -3045 -———— -- Internal Stuff - - Update the targeted .Net version to 4.5.2 (4.5 and 4.5.1 are no longer supported by Microsoft) - - Upgrade System.Net.FtpClient to replacement FluentFTP package - - Update Microsoft.Net.Http package to latest stable version - - Update fastJSON package to latest stable version - - Update embedIO package to latest stable version - - Remove Alchemy WS package from build - -- Davis TCP connections. - - Added additional error handling, should now be much more robust and attempt to reconnect on failure. - -- Astro calcs for Solar - - Added refraction correction. - - updated to add some extra terms. - -- Removed URL encode Twitter messages added in b3044 - -- Added second order humidity correction factor, works the same as the temperature. It has to be set manually in Cumulus.ini. - - Cumulus.ini - [Offsets] - HumMult2=0.0 - -- FTP Updates - - Now supports FTPS over TLS1.1 and 1.2 - - In Passive FTP, you can disable Enhanced Passive (EPSV) mode if it causes problems with your host. Some hosts are reporting they support it, but a firewall along the route cannot handle the connections. - Requires manually adding a new Cumulus.ini entry... - [FTP site] - DisableEPSV=1 - - RealtimeGaugesTxt is no longer automatically enabled for FTP. - -- Web Sockets are no longer on a separate port, it shares the same port as HTTP. - - The wsport command line switch is still recognised for backwards compatibility, but it is no longer used. - -- Fix Fine Offset with solar logger reading, now limited to the reduced number of logger entries available on solar stations. - -- Fix for Slovenian locale (and any other with a two character date separator) - - - -3044 -———— -- Added new solar calculation method "Bras" - - New Cumulus.ini entries... - SolarCalc=0 (0=Ryan-Stolzenbach, 1=Bras, default=0) - BrasTurbidity=2 (atmospheric turbidity factor (2=clear, 5=smoggy, default=2) - - Updated the Interface web files to reflect the new settings. - -- Updated HidSharp to ver 2.0.5 - - Now uses libudev1 for Fine Offset and WMR200 stations - -- Fixed corrupt/missing MySQL port causing CumulusMX to crash on start-up - -- Attempt to reconnect to Davis IP logger on connection failure (only a possible fix) - -- Updated the default forum URL to it's new home - -- Updated included website files to new URL - -- URL encode Twitter messages (I know Twitter is largely broken) - -- Davis VP2 - (Steve Loft) Added sanity checks for invalid wind speed/direction values - -- Davis VP2 - Added automatic disabling of the use of LOOP2 packets on firmware versions < 1.90 - -- Change RG11 devices to use a new Cumulus.ini file entry. The previous MX builds would not work on Linux. - For example: - RG11port=2 (depreciated) - RG11port2=3 - now use - RG11portName=COM2 (for Windows or /dev/ttyUSB2 for Linux) - RG11portName2=COM3 (for Windows or /dev/ttyUSB3 for Linux) - -- The default Comm port values are now set to either COM1 or /dev/ttyUSB0 depending on the platform. Hopefully this will prevent some of the confusion of new users. From 8da9da0c8864063e73f768771140132e5812199c Mon Sep 17 00:00:00 2001 From: Mark Crossley Date: Tue, 1 Oct 2024 15:52:54 +0100 Subject: [PATCH 63/63] Readme.md --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index 2b081056..6e1037f4 100644 --- a/README.md +++ b/README.md @@ -13,4 +13,4 @@ The original source was post build b3043 - it incorporated some changes that wer The support forum for this software can [be found here](https://cumulus.hosiene.co.uk/) -The change log [is here](blob/master/CHANGELOG.md) +The change log [is here](CHANGELOG.md)