diff --git a/CHANGELOG.md b/CHANGELOG.md
index 8bebaa7..95e77f4 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -2,6 +2,9 @@
All notable changes to this project will be documented in this file. This project uses [Semantic Versioning](https://semver.org/).
+## v4.0.0
+* Added Local UDP API support! Now you can choose to listen to your Weather Stations observations directly over your local network. No Station ID or API Token needed. Observations are broadcasted every 60 seconds. This leverages the `obs_st` message. See [documentation](https://weatherflow.github.io/Tempest/api/udp/v171/) for more information.
+
## v3.0.3
* Update node-version: [18.x, 20.x], remove 16.x which is no longer supported by homebridge.
* Reformated `getStationObservation()` and `getStationCurrentObservation()` in `tempestApi.ts`.
diff --git a/README.md b/README.md
index c052932..03fe8a4 100644
--- a/README.md
+++ b/README.md
@@ -9,6 +9,8 @@
+*New* in v4.0.0 Local API Support!
+
Homebridge Plugin providing basic WeatherFlow Tempest support. Exposing 7 Acessories.
- Temperature Sensor
@@ -34,9 +36,10 @@ It is recommended when upgrading to v3.0 of the plugin from a prior version that
You will need to create an account at https://tempestwx.com/ and then generate a Personal Use Token https://tempestwx.com/settings/tokens.
- `name`: _(Required)_ Must always be set to `WeatherFlow Tempest Platform`.
-- `token`: _(Required)_ Oauth2 Personal Use Token, create via your tempestwx account.
-- `station_id`: _(Required)_ The station ID you are pulling weather data from.
-- `interval`: _(Required)_ How often to poll the Tempest REST API. Default 10 seconds. Minimum every second.
+- `local_api`: _(Required)_ Use the Local API versus HTTP API.
+- `token`: _(Required for HTTP API)_ Oauth2 Personal Use Token, create via your tempestwx account.
+- `station_id`: _(Required for HTTP API)_ The station ID you are pulling weather data from.
+- `interval`: _(Required for HTTP API)_ How often to poll the Tempest REST API. Default 10 seconds. Minimum every second.
- `sensors`: _(Required)_ An array of sensors to create. This is dynamic incase you want to target different temperature or wind speed attributes.
- `sensors[].name`: _(Required)_ Display name of Sensor in Apple Home.
- `sensors[].sensor_type`: _(Required)_ The type of Home Sensor to create. There are 6 options ["Temperature Sensor", "Light Sensor", "Humidity Sensor", "Fan", "Motion Sensor", "Occupancy Sensor"].
@@ -69,11 +72,109 @@ sensor_type `{2}` | value_key | metric units | std units | additional_properties
`{4}` NOTE: There is a current limitation with v3.0.0 of the plug-in in that HomeKit accessory names are set when the accessory is initially added and cannot be dynamically updated. The accessories are correctly displayed and updated in the Homebridge "Accessories" tab of the webpage interface. Occupancy sensors `trigger_value` status is correctly displayed in both HomeKit and Homebridge.
-### Config Example
+### Local API Config Example
+
+```json
+{
+ "name": "WeatherFlow Tempest Platform",
+ "local_api": true,
+ "units": "Standard",
+ "sensors": [
+ {
+ "name": "Temperature",
+ "sensor_type": "Temperature Sensor",
+ "temperature_properties": {
+ "value_key": "air_temperature"
+ }
+ },
+ {
+ "name": "Relative Humidity",
+ "sensor_type": "Humidity Sensor",
+ "humidity_properties": {
+ "value_key": "relative_humidity"
+ }
+ },
+ {
+ "name": "Light Level",
+ "sensor_type": "Light Sensor",
+ "light_properties": {
+ "value_key": "brightness"
+ }
+ },
+ {
+ "name": "Wind Speed",
+ "sensor_type": "Fan",
+ "fan_properties": {
+ "value_key": "wind_avg"
+ }
+ },
+ {
+ "name": "Wind Gust",
+ "sensor_type": "Motion Sensor",
+ "motion_properties": {
+ "value_key": "wind_gust",
+ "trigger_value": 10
+ }
+ },
+ {
+ "name": "Barometer",
+ "sensor_type": "Occupancy Sensor",
+ "occupancy_properties": {
+ "value_key": "barometric_pressure",
+ "trigger_value": 30
+ }
+ },
+ {
+ "name": "Solar Radiation",
+ "sensor_type": "Occupancy Sensor",
+ "occupancy_properties": {
+ "value_key": "solar_radiation",
+ "trigger_value": 1000
+ }
+ },
+ {
+ "name": "UV",
+ "sensor_type": "Occupancy Sensor",
+ "occupancy_properties": {
+ "value_key": "uv",
+ "trigger_value": 3
+ }
+ },
+ {
+ "name": "Precipitation Rate",
+ "sensor_type": "Occupancy Sensor",
+ "occupancy_properties": {
+ "value_key": "precip",
+ "trigger_value": 0.25
+ }
+ },
+ {
+ "name": "Precipitation Today",
+ "sensor_type": "Occupancy Sensor",
+ "occupancy_properties": {
+ "value_key": "precip_accum_local_day",
+ "trigger_value": 1
+ }
+ },
+ {
+ "name": "Wind Direction",
+ "sensor_type": "Occupancy Sensor",
+ "occupancy_properties": {
+ "value_key": "wind_direction",
+ "trigger_value": 360
+ }
+ }
+ ],
+ "platform": "WeatherFlowTempest"
+}
+```
+
+### HTTP API Config Example
```json
{
"name": "WeatherFlow Tempest Platform",
+ "local_api": false,
"token": "XXXXXXXX-XXXX-XXXX-XXXX-XXXXXXXXXXXX",
"station_id": 10000,
"interval": 10,
diff --git a/config.schema.json b/config.schema.json
index ff4810f..f115681 100644
--- a/config.schema.json
+++ b/config.schema.json
@@ -1,182 +1,214 @@
{
- "pluginAlias": "WeatherFlowTempest",
- "pluginType": "platform",
- "singular": false,
- "schema": {
- "type": "object",
- "properties": {
- "name": {
- "title": "Name",
- "type": "string",
- "required": true,
- "default": "WeatherFlow Tempest Platform"
- },
- "token": {
- "title": "Token",
- "type": "string",
- "required": true
- },
- "station_id": {
- "title": "Station ID (Integer)",
- "type": "number",
- "required": true
- },
- "interval": {
- "title": "Interval (seconds)",
- "type": "integer",
- "default": 10,
- "minimum": 1
- },
- "units": {
- "title": "Units",
- "type": "string",
- "enum": [
- "Standard",
- "Metric"
- ],
- "default": "Standard"
- },
- "sensors": {
- "title": "Weather Sensors",
- "description": "Enable WeatherFlow Tempest Sensors.",
- "type": "array",
- "items": {
- "type": "object",
- "properties": {
- "name": {
- "title": "Name",
- "type": "string",
- "required": true
- },
- "sensor_type": {
- "type": "string",
- "enum": [
- "Temperature Sensor",
- "Light Sensor",
- "Humidity Sensor",
- "Fan",
- "Motion Sensor",
- "Occupancy Sensor"
- ],
- "default": "Temperature Sensor"
- },
- "fan_properties": {
- "title": "Fan Properties",
- "type": "object",
- "condition": {
- "functionBody": "if (model.sensors[arrayIndices] && model.sensors[arrayIndices].sensor_type && model.sensors[arrayIndices].sensor_type === 'Fan') { return true; } else { return false; };"
- },
- "properties": {
- "value_key": {
- "type": "string",
- "enum": [
- "wind_avg"
- ]
- }
- }
- },
- "light_properties": {
- "title": "Light Properties",
- "type": "object",
- "condition": {
- "functionBody": "if (model.sensors[arrayIndices] && model.sensors[arrayIndices].sensor_type && model.sensors[arrayIndices].sensor_type === 'Light Sensor') { return true; } else { return false; };"
- },
- "properties": {
- "value_key": {
- "type": "string",
- "enum": [
- "brightness"
- ]
- }
- }
- },
- "humidity_properties": {
- "title": "Humidity Properties",
- "type": "object",
- "condition": {
- "functionBody": "if (model.sensors[arrayIndices] && model.sensors[arrayIndices].sensor_type && model.sensors[arrayIndices].sensor_type === 'Humidity Sensor') { return true; } else { return false; };"
- },
- "properties": {
- "value_key": {
- "type": "string",
- "enum": [
- "relative_humidity"
- ]
- }
- }
- },
- "temperature_properties": {
- "title": "Temperature Properties",
- "type": "object",
- "condition": {
- "functionBody": "if (model.sensors[arrayIndices] && model.sensors[arrayIndices].sensor_type && model.sensors[arrayIndices].sensor_type === 'Temperature Sensor') { return true; } else { return false; };"
- },
- "properties": {
- "value_key": {
- "type": "string",
- "enum": [
- "air_temperature",
- "dew_point",
- "feels_like",
- "wind_chill"
- ]
- }
- }
- },
- "motion_properties": {
- "title": "Motion Properties",
- "type": "object",
- "condition": {
- "functionBody": "if (model.sensors[arrayIndices] && model.sensors[arrayIndices].sensor_type && model.sensors[arrayIndices].sensor_type === 'Motion Sensor') { return true; } else { return false; };"
- },
- "properties": {
- "value_key": {
- "type": "string",
- "enum": [
- "wind_gust"
- ]
- },
- "trigger_value": {
- "type": "number",
- "minimum": 1,
- "description": "At what point (value) to trigger motion detected on/off (1 minimum)."
- }
- }
- },
- "occupancy_properties": {
- "title": "Occupancy Properties",
- "type": "object",
- "condition": {
- "functionBody": "if (model.sensors[arrayIndices] && model.sensors[arrayIndices].sensor_type && model.sensors[arrayIndices].sensor_type === 'Occupancy Sensor') { return true; } else { return false; };"
- },
- "properties": {
- "value_key": {
- "type": "string",
- "enum": [
- "barometric_pressure",
- "precip",
- "precip_accum_local_day",
- "wind_direction",
- "wind_gust",
- "solar_radiation",
- "uv"
- ]
- },
- "trigger_value": {
- "type": "number",
- "minimum": 0,
- "description": "At what point (value) to trigger occupancy detected on/off (0 minimum)."
- }
- }
- }
- }
- },
- "required": [
- "name",
- "sensor_type",
- "value_key"
- ]
- }
- }
- }
+ "pluginAlias": "WeatherFlowTempest",
+ "pluginType": "platform",
+ "singular": false,
+ "schema": {
+ "type": "object",
+
+ "properties": {
+ "name": {
+ "title": "Name",
+ "type": "string",
+ "required": true,
+ "default": "WeatherFlow Tempest Platform"
+ },
+ "local_api": {
+ "title": "Use Local API",
+ "type": "boolean",
+ "required": true,
+ "default": true
+ },
+ "token": {
+ "title": "Token",
+ "type": "string",
+ "default": "",
+ "condition": {
+ "functionBody": "if (model.local_api != undefined && !model.local_api) { return true; } else { return false; };"
+ }
+ },
+ "station_id": {
+ "title": "Station ID (Integer)",
+ "type": "number",
+ "default": 0,
+ "condition": {
+ "functionBody": "if (model.local_api != undefined && !model.local_api) { return true; } else { return false; };"
+ }
+ },
+ "interval": {
+ "title": "Interval (seconds)",
+ "type": "integer",
+ "default": 10,
+ "minimum": 1,
+ "condition": {
+ "functionBody": "if (model.local_api != undefined && !model.local_api) { return true; } else { return false; };"
+ }
+ },
+ "units": {
+ "title": "Units",
+ "type": "string",
+ "enum": [
+ "Standard",
+ "Metric"
+ ],
+ "default": "Standard"
+ },
+ "sensors": {
+ "title": "Weather Sensors",
+ "description": "Enable WeatherFlow Tempest Sensors.",
+ "type": "array",
+ "items": {
+ "type": "object",
+ "properties": {
+ "name": {
+ "title": "Name",
+ "type": "string",
+ "required": true
+ },
+ "sensor_type": {
+ "type": "string",
+ "enum": [
+ "Temperature Sensor",
+ "Light Sensor",
+ "Humidity Sensor",
+ "Fan",
+ "Motion Sensor",
+ "Occupancy Sensor"
+ ],
+ "default": "Temperature Sensor"
+ },
+ "fan_properties": {
+ "title": "Fan Properties",
+ "type": "object",
+ "condition": {
+ "functionBody": "if (model.sensors[arrayIndices] && model.sensors[arrayIndices].sensor_type && model.sensors[arrayIndices].sensor_type === 'Fan') { return true; } else { return false; };"
+ },
+ "properties": {
+ "value_key": {
+ "type": "string",
+ "enum": [
+ "wind_avg"
+ ]
+ }
+ }
+ },
+ "light_properties": {
+ "title": "Light Properties",
+ "type": "object",
+ "condition": {
+ "functionBody": "if (model.sensors[arrayIndices] && model.sensors[arrayIndices].sensor_type && model.sensors[arrayIndices].sensor_type === 'Light Sensor') { return true; } else { return false; };"
+ },
+ "properties": {
+ "value_key": {
+ "type": "string",
+ "enum": [
+ "brightness"
+ ]
+ }
+ }
+ },
+ "humidity_properties": {
+ "title": "Humidity Properties",
+ "type": "object",
+ "condition": {
+ "functionBody": "if (model.sensors[arrayIndices] && model.sensors[arrayIndices].sensor_type && model.sensors[arrayIndices].sensor_type === 'Humidity Sensor') { return true; } else { return false; };"
+ },
+ "properties": {
+ "value_key": {
+ "type": "string",
+ "enum": [
+ "relative_humidity"
+ ]
+ }
+ }
+ },
+ "temperature_properties": {
+ "title": "Temperature Properties",
+ "type": "object",
+ "condition": {
+ "functionBody": "if (model.sensors[arrayIndices] && model.sensors[arrayIndices].sensor_type && model.sensors[arrayIndices].sensor_type === 'Temperature Sensor') { return true; } else { return false; };"
+ },
+ "properties": {
+ "value_key": {
+ "type": "string",
+ "enum": [
+ "air_temperature",
+ "dew_point",
+ "feels_like",
+ "wind_chill"
+ ]
+ }
+ }
+ },
+ "motion_properties": {
+ "title": "Motion Properties",
+ "type": "object",
+ "condition": {
+ "functionBody": "if (model.sensors[arrayIndices] && model.sensors[arrayIndices].sensor_type && model.sensors[arrayIndices].sensor_type === 'Motion Sensor') { return true; } else { return false; };"
+ },
+ "properties": {
+ "value_key": {
+ "type": "string",
+ "enum": [
+ "wind_gust"
+ ],
+ "condition": {
+ "functionBody": "if (!model.local_api && model.sensors[arrayIndices] && model.sensors[arrayIndices].sensor_type && model.sensors[arrayIndices].sensor_type === 'Temperature Sensor') { return true; } else { return false; };"
+ }
+ },
+ "trigger_value": {
+ "type": "number",
+ "minimum": 1,
+ "description": "At what point (value) to trigger motion detected on/off (1 minimum)."
+ }
+ }
+ },
+ "occupancy_properties": {
+ "title": "Occupancy Properties",
+ "type": "object",
+ "condition": {
+ "functionBody": "if (model.sensors[arrayIndices] && model.sensors[arrayIndices].sensor_type && model.sensors[arrayIndices].sensor_type === 'Occupancy Sensor') { return true; } else { return false; };"
+ },
+ "properties": {
+ "value_key": {
+ "type": "string",
+ "enum": [
+ "barometric_pressure",
+ "precip",
+ "precip_accum_local_day",
+ "wind_direction",
+ "wind_gust",
+ "solar_radiation",
+ "uv"
+ ],
+ "condition": {
+ "functionBody": "if (model.sensors[arrayIndices] && model.sensors[arrayIndices].sensor_type && model.sensors[arrayIndices].sensor_type === 'Temperature Sensor') { return true; } else { return false; };"
+ }
+ },
+ "trigger_value": {
+ "type": "number",
+ "minimum": 0,
+ "description": "At what point (value) to trigger occupancy detected on/off (0 minimum)."
+ }
+ }
+ }
+ }
+ },
+ "required": [
+ "name",
+ "sensor_type",
+ "value_key"
+ ]
+ }
+ },
+ "required": ["local_api"],
+ "dependencies": {
+ "local_api": {
+ "not": {
+ "type": "boolean",
+ "const": true
+ },
+ "required": ["token", "station_id"]
+ }
+ }
+ }
}
diff --git a/package-lock.json b/package-lock.json
index ac6a6d7..bd5eeae 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -1,12 +1,12 @@
{
"name": "homebridge-weatherflow-tempest",
- "version": "3.0.3",
+ "version": "4.0.0",
"lockfileVersion": 3,
"requires": true,
"packages": {
"": {
"name": "homebridge-weatherflow-tempest",
- "version": "3.0.3",
+ "version": "4.0.0",
"license": "Apache-2.0",
"dependencies": {
"axios": "1.5.1"
diff --git a/package.json b/package.json
index 134732c..37665ee 100644
--- a/package.json
+++ b/package.json
@@ -2,7 +2,7 @@
"private": false,
"displayName": "Homebridge WeatherFlow Tempest",
"name": "homebridge-weatherflow-tempest",
- "version": "3.0.3",
+ "version": "4.0.0",
"description": "Exposes WeatherFlow Tempest Station data as Temperature Sensors, Light Sensors, Humidity Sensors and Fan Sensors (for Wind Speed).",
"license": "Apache-2.0",
"repository": {
diff --git a/src/platform.ts b/src/platform.ts
index 5880614..f5dd0a1 100644
--- a/src/platform.ts
+++ b/src/platform.ts
@@ -3,7 +3,7 @@ import { API, DynamicPlatformPlugin, Logger, PlatformAccessory, PlatformConfig,
import { PLATFORM_NAME, PLUGIN_NAME } from './settings';
import { WeatherFlowTempestPlatformAccessory } from './platformAccessory';
-import { TempestApi, Observation } from './tempestApi';
+import { TempestApi, TempestSocket, Observation } from './tempest';
interface TempestSensor {
name: string;
@@ -24,7 +24,8 @@ export class WeatherFlowTempestPlatform implements DynamicPlatformPlugin {
public readonly Characteristic: typeof Characteristic = this.api.hap.Characteristic;
public readonly accessories: PlatformAccessory[] = [];
- private tempestApi: TempestApi;
+ private tempestApi: TempestApi | undefined;
+ private tempestSocket: TempestSocket | undefined;
public observation_data: Observation; // Observation data for Accessories to use.
public tempest_battery_level!: number; // Tempest battery level
@@ -40,10 +41,7 @@ export class WeatherFlowTempestPlatform implements DynamicPlatformPlugin {
log.info('Finished initializing platform:', this.config.name);
- // Initialize TempestApi
- this.tempestApi = new TempestApi(this.config.token, this.config.station_id, log);
-
- // initialize observation_data
+ // Initialize observation_data
this.observation_data = {
air_temperature: 0,
barometric_pressure: 0,
@@ -64,7 +62,7 @@ export class WeatherFlowTempestPlatform implements DynamicPlatformPlugin {
this.tempest_device_id = 0;
// Make sure the Station ID is the integer ID
- if (isNaN(this.config.station_id)) {
+ if (this.config.local_api === false && isNaN(this.config.station_id)) {
log.warn(
'Station ID is not an Integer! Please make sure you are using the ID integer found here: ' +
'https://tempestwx.com/station//',
@@ -81,56 +79,125 @@ export class WeatherFlowTempestPlatform implements DynamicPlatformPlugin {
return;
}
- try {
+ // Initialize Tempest Interfaces
+ if (this.config.local_api === true) {
+ this.initializeBySocket();
+ } else {
+ this.initializeByApi();
+ }
- this.tempestApi.getStationCurrentObservation(0).then( (observation_data: Observation) => {
+ });
- if (!observation_data) {
- log.warn('Failed to fetch initial Station Current Observations after retrying. Refusing to continue.');
- return;
- }
+ }
- // Cache the observation results
- this.observation_data = observation_data;
+ private async initializeBySocket() {
+
+ try {
+ this.log.info('Using Tempest Local API.');
+ this.tempestSocket = new TempestSocket(this.log);
+ this.tempestSocket.start();
+
+ // Hold thread for first message and set values
+ await this.socketDataRecieved();
+ this.observation_data = this.tempestSocket.getStationCurrentObservation();
+ this.tempest_battery_level = this.tempestSocket.getBatteryLevel();
+
+ // Initialize sensors after first API response.
+ this.discoverDevices();
+ this.log.debug ('discoverDevices completed');
- // Initialize sensors after first API response.
- this.discoverDevices();
+ // Poll every minute for local API
+ this.pollLocalStationCurrentObservation();
- this.log.debug ('discoverDevices completed');
- // Remove cached sensors that are no longer required.
- this.removeDevices();
+ } catch(exception) {
+ this.log.error(exception as string);
+ }
+ }
- this.log.debug ('removeDevices completed');
+ private socketDataRecieved(): Promise {
- // Determine Tempest device_id & initial battery level
- this.tempestApi.getTempestDeviceId().then( (device_id: number) => {
- this.tempest_device_id = device_id;
+ this.log.info('Waiting for first local broadcast. This could take up to 60 seconds...');
+ return new Promise((resolve) => {
+ const socket_interval = setInterval(() => {
+ if (this.tempestSocket === undefined) {
+ return;
+ }
+ if (this.tempestSocket.hasData()) {
+ clearInterval(socket_interval);
+ this.log.info('Initial local broadcast recieved.');
+ resolve();
+ }
+ }, 1000);
+ });
- this.tempestApi.getTempestBatteryLevel(this.tempest_device_id).then( (battery_level: number) => {
+ }
- if (battery_level === undefined) {
- this.log.warn('Failed to fetch initial Tempest battery level');
- return;
- }
- this.tempest_battery_level = battery_level;
+ private initializeByApi() {
- });
+ try {
+ this.log.info('Using Tempest RESTful API.');
+ this.tempestApi = new TempestApi(this.config.token, this.config.station_id, this.log);
+ this.tempestApi.getStationCurrentObservation(0).then( (observation_data: Observation) => {
- });
+ if (!observation_data) {
+ this.log.warn('Failed to fetch initial Station Current Observations after retrying. Refusing to continue.');
+ return;
+ }
- // Then begin to poll the station current observations data.
- this.pollStationCurrentObservation();
+ if (this.tempestApi === undefined) {
+ return;
+ }
+
+ // Cache the observation results
+ this.observation_data = observation_data;
+
+ // Initialize sensors after first API response.
+ this.discoverDevices();
+ this.log.debug ('discoverDevices completed');
+ // Remove cached sensors that are no longer required.
+ this.removeDevices();
+ this.log.debug ('removeDevices completed');
+
+ // Determine Tempest device_id & initial battery level
+ this.tempestApi.getTempestDeviceId().then( (device_id: number) => {
+ this.tempest_device_id = device_id;
+ if (this.tempestApi === undefined) {
+ return;
+ }
+ this.tempestApi.getTempestBatteryLevel(this.tempest_device_id).then( (battery_level: number) => {
+ if (battery_level === undefined) {
+ this.log.warn('Failed to fetch initial Tempest battery level');
+ return;
+ }
+ this.tempest_battery_level = battery_level;
+ });
});
- } catch(exception) {
+ // Then begin to poll the station current observations data.
+ this.pollStationCurrentObservation();
+ });
+
+ } catch(exception) {
+ this.log.error(exception as string);
+ }
+
+ }
+
+ private pollLocalStationCurrentObservation(): void {
- this.log.error(exception as string);
+ setInterval( async () => {
+ if (this.tempestSocket === undefined) {
+ return;
}
- });
+ // Update values
+ this.observation_data = this.tempestSocket.getStationCurrentObservation();
+ this.tempest_battery_level = this.tempestSocket.getBatteryLevel();
+
+ }, 60 * 1000); // Tempest local API broadcasts every minute.
}
@@ -142,6 +209,10 @@ export class WeatherFlowTempestPlatform implements DynamicPlatformPlugin {
setInterval( async () => {
+ if (this.tempestApi === undefined) {
+ return;
+ }
+
// Update Observation data
await this.tempestApi.getStationCurrentObservation(0).then( (observation_data: Observation) => {
diff --git a/src/platformAccessory.ts b/src/platformAccessory.ts
index 964f0fd..57b819b 100644
--- a/src/platformAccessory.ts
+++ b/src/platformAccessory.ts
@@ -302,16 +302,27 @@ class Fan {
try {
const value_key: string = this.accessory.context.device.fan_properties.value_key;
let speed: number = parseFloat(this.platform.observation_data[value_key]);
- speed = Math.round(speed * 2.236936); // convert m/s to mph and round as fan % is integer value
- if (speed > 100) {
- this.platform.log.debug(`WeatherFlow Tempest is reporting wind speed exceeding 100mph: ${speed}mph`);
- return 100;
- } else if (speed < 0) {
- this.platform.log.debug(`WeatherFlow Tempest is reporting wind speed less than 0mph: ${speed}mph`);
- return 0;
+ if (this.platform.config.units === 'Metric') {
+ speed = Math.round(speed); // round as fan % is integer value
+ if (speed > 45) {
+ this.platform.log.debug(`WeatherFlow Tempest is reporting wind speed exceeding 45 m/s: ${speed} m/s`);
+ speed = 45;
+ } else if (speed < 0) {
+ this.platform.log.debug(`WeatherFlow Tempest is reporting wind speed less than 0 m/s: ${speed} m/s`);
+ speed = 0;
+ }
} else {
- return speed;
+ speed = Math.round(speed * 2.236936); // convert m/s to mph and round as fan % is integer value
+ if (speed > 100) {
+ this.platform.log.debug(`WeatherFlow Tempest is reporting wind speed exceeding 100 mph: ${speed} mph`);
+ speed = 100;
+ } else if (speed < 0) {
+ this.platform.log.debug(`WeatherFlow Tempest is reporting wind speed less than 0 mph: ${speed} mph`);
+ speed = 0;
+ }
}
+ return speed;
+
} catch(exception) {
this.platform.log.error(exception as string);
return 0;
diff --git a/src/tempestApi.ts b/src/tempest.ts
similarity index 57%
rename from src/tempestApi.ts
rename to src/tempest.ts
index 32606fb..5653ade 100644
--- a/src/tempestApi.ts
+++ b/src/tempest.ts
@@ -1,10 +1,13 @@
import { Logger } from 'homebridge';
import axios, { AxiosResponse } from 'axios';
+import * as dgram from 'dgram';
import https from 'https';
+
axios.defaults.timeout = 10000; // same as default interval
axios.defaults.httpsAgent = new https.Agent({ keepAlive: true });
+
export interface Observation {
// temperature sensors
air_temperature: number; // C, displayed according to Homebridge and HomeKit C/F settings
@@ -30,6 +33,125 @@ export interface Observation {
brightness: number; // Lux
}
+
+export class TempestSocket {
+
+ private log: Logger;
+ private s: dgram.Socket;
+ private data: object | undefined;
+ private tempest_battery_level: number;
+
+ constructor(log: Logger) {
+
+ this.log = log;
+ this.data = undefined;
+ this.tempest_battery_level = 0;
+ this.s = dgram.createSocket('udp4');
+
+ this.log.info('TempestSocket initialized.');
+
+ }
+
+ public start(address = '0.0.0.0', port = 50222) {
+
+ this.setupSocket(address, port);
+ this.setupSignalHandlers();
+
+ }
+
+ private setupSocket(address: string, port: number) {
+
+ this.s.bind({ address: address, port: port });
+ this.s.on('message', (msg) => {
+ try {
+ const message_string = msg.toString('utf-8');
+ const data = JSON.parse(message_string);
+ this.processReceivedData(data);
+ } catch (error) {
+ this.log.warn('JSON processing of data failed');
+ this.log.error(error as string);
+ }
+ });
+
+ this.s.on('error', (err) => {
+ this.log.error('Socket error:', err);
+ });
+
+ }
+
+ private processReceivedData(data) {
+
+ if (data.type === 'obs_st') { // for Tempest
+ this.setTempestData(data);
+ }
+
+ }
+
+ private setTempestData(data): void {
+
+ const obs = data.obs[0];
+ // const windLull = (obs[1] !== null) ? obs[1] : 0;
+ const windSpeed = (obs[2] !== null) ? obs[2] * 2.2369 : 0; // convert to mph for heatindex calculation
+ // const windGust = (obs[3] !== null) ? Math.round(obs[3] * 2.236936) : 0; // convert to mph
+ const T = (obs[7] * 9/5) + 32; // T in F for heatindex, feelsLike and windChill calculations
+
+ // eslint-disable-next-line max-len
+ const heatIndex = -42.379 + 2.04901523*T + 10.14333127*obs[8] - 0.22475541*T*obs[8] - 0.00683783*(T**2) - 0.05481717*(obs[8]**2) + 0.00122874*(T**2)*obs[8] + 0.00085282*T*(obs[8]**2) - 0.00000199*(T**2)*(obs[8]**2);
+
+ // feels like temperature on defined for temperatures between 80F and 110F
+ const feelsLike = ((T >= 80) && (T <= 110)) ? heatIndex : T;
+
+ // windChill only defined for wind speeds > 3 mph and temperature < 50F
+ const windChill = ((windSpeed > 3) && (T < 50)) ? (35.74 + 0.6215*T - 35.75*(windSpeed**0.16) + 0.4275*T*(windSpeed**0.16)) : T;
+
+ this.data = {
+ air_temperature: obs[7],
+ feels_like: 5/9 * (feelsLike - 32), // convert back to C
+ wind_chill: 5/9 * (windChill - 32), // convert back to C
+ dew_point: obs[7] - ((100 - obs[8]) / 5.0), // Td = T - ((100 - RH)/5)
+ relative_humidity: obs[8],
+ wind_avg: obs[2],
+ wind_gust: obs[3],
+ barometric_pressure: obs[6],
+ precip: obs[12],
+ precip_accum_local_day: obs[12],
+ wind_direction: obs[4],
+ solar_radiation: obs[11],
+ uv: obs[10],
+ brightness: obs[9],
+ };
+ this.tempest_battery_level = Math.round((obs[16] - 1.8) * 100); // 2.80V = 100%, 1.80V = 0%
+
+ }
+
+ private setupSignalHandlers(): void {
+
+ process.on('SIGTERM', () => {
+ this.log.info('Got SIGTERM, shutting down Tempest Homebridge...');
+ this.s.close();
+ });
+
+ process.on('SIGINT', () => {
+ this.log.info('Got SIGINT, shutting down Tempest Homebridge...');
+ this.s.close();
+ });
+
+ }
+
+ public hasData(): boolean {
+ return this.data !== undefined;
+ }
+
+ public getStationCurrentObservation(): Observation {
+ return this.data as Observation;
+ }
+
+ public getBatteryLevel(): number {
+ return this.tempest_battery_level;
+ }
+
+}
+
export class TempestApi {
private log: Logger;