[metOfficeDataHub] Initial contribution (#15367)

* [metOfficeDataHub] Initial Commit for v4

[metOfficeDataHub] Initial code commit.

Signed-off-by: David Goodyear <david.goodyear@gmail.com>
This commit is contained in:
dag81 2024-11-22 08:40:51 +00:00 committed by GitHub
parent e607a9934d
commit 38b04943cc
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
40 changed files with 7107 additions and 0 deletions

View File

@ -1086,6 +1086,11 @@
<artifactId>org.openhab.binding.meteostick</artifactId>
<version>${project.version}</version>
</dependency>
<dependency>
<groupId>org.openhab.addons.bundles</groupId>
<artifactId>org.openhab.binding.metofficedatahub</artifactId>
<version>${project.version}</version>
</dependency>
<dependency>
<groupId>org.openhab.addons.bundles</groupId>
<artifactId>org.openhab.binding.mffan</artifactId>

View File

@ -0,0 +1,13 @@
This content is produced and maintained by the openHAB project.
* Project home: https://www.openhab.org
== Declared Project Licenses
This program and the accompanying materials are made available under the terms
of the Eclipse Public License 2.0 which is available at
https://www.eclipse.org/legal/epl-2.0/.
== Source Code
https://github.com/openhab/openhab-addons

View File

@ -0,0 +1,446 @@
# Met Office DataHub Binding
This binding is for the UK Based Met Office Data Hub, weather service.
Its purpose is to allow the retrieval of forecast (hourly and daily) for a given location (Site).
The website can be found here: <https://datahub.metoffice.gov.uk/>
**IMPORTANT:** The Met Office Data Hub service is free of charge for low volume users.
Higher data usages are charged, please see their website for current information.
Please bear this in mind before adjust polling rates, or adding more than 1 location (site) for forecast data, as you may need a different plan depending on the data throughput over a month, or API hit rate.
A possible use case could be to pull forecast data, for the next day to determine if storage heaters or underfloor heating should be pre-heated overnight.
## Prerequisite
In order to use this binding, you will need a Met Office Data Hub account.
Once created you will need to create a plan for access to the "Site Specific" subscriptions.
This will give you the client id and secret required for the bridge.
## Supported Things
This binding consists of a bridge for connecting to the Met Office Data Hub service with your account.
You can then add things to get the forecast's for a specific location (site), using this bridge.
This binding supports the follow thing types:
| Type UID | Discovery | Description |
|-----------|-----------|---------------------------------------------------------------------------------------------|
| bridge | Manual | A single connection to the Met Office DataHub API with daily poll limiting for the Site API |
| site | Manual | Provides the hourly and daily forecast data for a give location (site) |
## Configuration
### `bridge` Configuration
The bridge counts the total number of requests from 00:00 -> 23:59 under its properties during the runtime of the system.
(This reset's if OH restarts, or the binding resets).
| Name | Type | Description | Default Values |
|--------------------|--------|-------------------------------------------------------------------------------------------|----------------|
| siteRateDailyLimit | Number | This is a daily poll limit for the SiteSpecific API, while the Thing ID remains the same. | 250 |
| siteApiKey | String | The API Key for the Site Specific subscription in your MET Office Data Hub account. | |
**NOTE:** siteRateDailyLimit: This **should** prevent any more poll's for the rest of the day to the SiteSpecific API, once this limit is reached as a failsafe against a bad configuration, if you don't reboot / delete and re-add the bridge. This is reset at 00:00UTC in-line with MET Office DataHub behaviours.
### `site` Configuration Parameters
| Name | Type | Description | Default Values |
|--------------------------|--------|----------------------------------------------------------------|-------------------------------------------------------|
| hourlyForecastPollRate | Number | The number of hours between polling for each sites hourly data | 1 |
| dailyForecastPollRate | Number | The number of hours between polling for each sites daily data | 3 |
| location | String | The lat/long of the site e.g. "51.5072,0.1276" | openHAB's user configured location is used when unset |
## Channels
### Hourly Forecast Channels
| Channel Id | Type | Description | Unit |
|------------------|----------------------|----------------------------------------------|------|
| forecast-ts | String | Time of forecast window start | |
| air-temp-current | Number:Temperature | Air Temperature | °C |
| air-temp-min | Number:Temperature | Minimum Air Temperature Over Previous Hour | °C |
| air-temp-max | Number:Temperature | Maximum Air Temperature Over Previous Hour | °C |
| feels-like | Number:Temperature | Feels Like Temperature | °C |
| humidity | Number:Dimensionless | Relative Humidity | % |
| visibility | Number:Length | Visibility | m |
| precip-rate | Number:Speed | Precipitation Rate | mm/h |
| precip-prob | Number:Dimensionless | Probability of Precipitation | % |
| precip-total | Number:Length | Total Precipitation of Previous Hour | mm |
| snow-total | Number:Length | Total Snowfall of Previous Hour | mm |
| uv-index | Number:Dimensionless | UV Index | |
| pressure | Number:Pressure | Mean Sea Level Pressure | Pa |
| wind-speed | Number:Speed | 10m Wind Speed | m/s |
| wind-gust | Number:Speed | 10m Wind Gust Speed | m/s |
| wind-gust-max | Number:Speed | Maximum 10m Wind Gust Speed of Previous Hour | m/s |
| wind-direction | Number:Angle | 10m Wind From Direction | ° |
| dewpoint | Number:Temperature | Dew Point Temperature | °C |
This binding uses channel groups.
The channels under "Forecast for the current hour" will be mirrored for future hours forecasts.
The channel naming follows the following format:
```current-forecast<Optional Offset Id>#air-temp-current```
The current hours forecast to get the air-temp-current would be:
current-forecast#air-temp-current
1 hour into the future to get the air-temp-current it would be:
current-forecast-**plus01**#air-temp-current
2 hour's into the future to get the air-temp-current it would be:
current-forecast-**plus02**#air-temp-current
#### Channel Groups for Hourly Forecast Channels
| Channel Id | Description |
|-------------------------|-------------------------------------------|
| current-forecast | Current hours forecast |
| current-forecast-plus01 | 01 hour after the current hours forecast |
| current-forecast-plus02 | 02 hours after the current hours forecast |
| ....................... | ......................................... |
| current-forecast-plus23 | 23 hours after the current hours forecast |
| current-forecast-plus24 | 24 hours after the current hours forecast |
### Daily Forecast Channels
| Channel Id | Type | Unit | MET Office Data Description |
|-------------------------|----------------------|------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| forecast-ts | String | | Calculated from the MET provided UTZ time of when the forecast is applicable, mapped to the local system TZ. |
| wind-speed-day | Number:Speed | m/s | Mean wind speed is equivalent to the mean speed observed over the 10 minutes preceding the validity time. 10m wind is the considered surface wind. |
| wind-speed-night | Number:Speed | m/s | Mean wind speed is equivalent to the mean speed observed over the 10 minutes preceding the validity time. 10m wind is the considered surface wind. |
| wind-direction-day | Number:Angle | ° | Mean wind direction is equivalent to the mean direction observed over the 10 minutes preceding the validity time. In meteorological reports the direction of the wind vector is given as the direction from which it is blowing. 10m wind is the considered surface wind. |
| wind-direction-night | Number:Angle | ° | Mean wind direction is equivalent to the mean direction observed over the 10 minutes preceding the validity time. In meteorological reports the direction of the wind vector is given as the direction from which it is blowing. 10m wind is the considered surface wind. |
| wind-gust-day | Number:Speed | m/s | The gust speed is equivalent to the maximum 3 second mean wind speed observed over the 10 minutes preceding the validity time. 10m wind is the considered surface wind. |
| wind-gust-night | Number:Speed | m/s | The gust speed is equivalent to the maximum 3 second mean wind speed observed over the 10 minutes preceding the validity time. 10m wind is the considered surface wind. |
| visibility-day | Number:Length | m | Minimal horizontal distance at which a known object can be seen. | |
| visibility-night | Number:Length | m | Minimal horizontal distance at which a known object can be seen. | |
| humidity-day | Number:Dimensionless | % | Stevenson screen height is approximately 1.5m above ground level. | |
| humidity-night | Number:Dimensionless | % | Stevenson screen height is approximately 1.5m above ground level. | |
| pressure-day | Number:Pressure | Pa | Air pressure at mean sea level which is close to the geoid in sea areas. Air pressure at sea level is the quantity often abbreviated as pressure or PMSL. |
| pressure-night | Number:Pressure | Pa | Air pressure at mean sea level which is close to the geoid in sea areas. Air pressure at sea level is the quantity often abbreviated as pressure or PMSL. |
| uv-max | Number:Dimensionless | | Usually a value from 0 to 13 but higher values are possible in extreme situations. Daytime is defined as those forecast times that fall between local dawn and dusk. |
| temp-max-day | Number:Temperature | °C | This is the most likely maximum value over the day based on the ensemble spread. Stevenson screen height is approximately 1.5m above ground level. Daytime is defined as those forecast times that fall between local dawn and dusk. |
| temp-min-night | Number:Temperature | °C | This is the most likely minimum value over the night based on the ensemble spread. Stevenson screen height is approximately 1.5m above ground level. Night-time is defined as those forecast times that fall between local dusk and dawn. |
| temp-max-lb-day | Number:Temperature | °C | This is the lower bound for the maximum value over the day based on the ensemble spread. It is actually given by the 2.5 percentile. This means there is a 97.5% probability that the actual figure will be above this lower bound figure. Stevenson screen height is approximately 1.5m above ground level. Daytime is defined as those forecast times that fall between local dawn and dusk. |
| temp-min-lb-night | Number:Temperature | °C | This is the lower bound for the minimum value over the night based on the ensemble spread. It is actually given by the 2.5 percentile. This means there is a 97.5% probability that the actual figure will be above this lower bound figure. Stevenson screen height is approximately 1.5m above ground level. Night-time is defined as those forecast times that fall between local dusk and dawn. |
| temp-max-ub-day | Number:Temperature | °C | This is the upper bound for the maximum value over the day based on the ensemble spread. It is actually given by the 97.5 percentile. This means there is a 97.5% probability that the actual figure will be below this upper bound figure. Stevenson screen height is approximately 1.5m above ground level. Daytime is defined as those forecast times that fall between local dawn and dusk. |
| temp-min-ub-night | Number:Temperature | °C | This is the upper bound for the minimum value over the night based on the ensemble spread. It is actually given by the 97.5 percentile. This means there is a 97.5% probability that the actual figure will be below this upper bound figure. Stevenson screen height is approximately 1.5m above ground level. Night-time is defined as those forecast times that fall between local dusk and dawn. |
| feels-like-max-day | Number:Temperature | °C | This is the most likely maximum value over the day based on the ensemble spread. This is the temperature it feels like taking into account humidity and wind chill but not radiation. Daytime is defined as those forecast times that fall between local dawn and dusk. |
| feels-like-min-night | Number:Temperature | °C | This is the most likely minimum value over the night based on the ensemble spread. This is the temperature it feels like taking into account humidity and wind chill but not radiation. Night-time is defined as those forecast times that fall between local dusk and dawn. |
| feels-like-max-lb-day | Number:Temperature | °C | This is the lower bound for the maximum value over the day based on the ensemble spread. It is actually given by the 2.5 percentile. This means there is a 97.5% probability that the actual figure will be above this lower bound figure. This is the temperature it feels like taking into account humidity and wind chill but not radiation. Daytime is defined as those forecast times that fall between local dawn and dusk. |
| feels-like-min-lb-night | Number:Temperature | °C | This is the lower bound for the minimum value over the night based on the ensemble spread. It is actually given by the 2.5 percentile. This means there is a 97.5% probability that the actual figure will be above this lower bound figure. This is the temperature it feels like taking into account humidity and wind chill but not radiation. Night-time is defined as those forecast times that fall between local dusk and dawn. |
| feels-like-max-ub-day | Number:Temperature | °C | This is the upper bound for the maximum value over the day based on the ensemble spread. It is actually given by the 97.5 percentile. This means there is a 97.5% probability that the actual figure will be below this upper bound figure. This is the temperature it feels like taking into account humidity and wind chill but not radiation. Daytime is defined as those forecast times that fall between local dawn and dusk. |
| feels-like-min-ub-night | Number:Temperature | °C | This is the upper bound for the minimum value over the night based on the ensemble spread. It is actually given by the 97.5 percentile. This means there is a 97.5% probability that the actual figure will be below this upper bound figure. This is the temperature it feels like taking into account humidity and wind chill but not radiation. Night-time is defined as those forecast times that fall between local dusk and dawn. |
| precip-prob-day | Number:Dimensionless | % | Daytime is defined as those forecast times that fall between local dawn and dusk. |
| precip-prob-night | Number:Dimensionless | % | Night-time is defined as those forecast times that fall between local dusk and dawn. |
| snow-prob-day | Number:Dimensionless | % | Daytime is defined as those forecast times that fall between local dawn and dusk. |
| snow-prob-night | Number:Dimensionless | % | Night-time is defined as those forecast times that fall between local dusk and dawn. |
| heavy-snow-prob-day | Number:Dimensionless | % | Heavy snow is defined as >1mm/hr liquid water equivalent and is approximately equivilent to >1cm snow per hour. Daytime is defined as those forecast times that fall between local dawn and dusk. |
| heavy-snow-prob-night | Number:Dimensionless | % | Heavy snow is defined as >1mm/hr liquid water equivalent and is approximately equivilent to >1cm snow per hour. Night-time is defined as those forecast times that fall between local dusk and dawn. |
| rain-prob-day | Number:Dimensionless | % | Daytime is defined as those forecast times that fall between local dawn and dusk. |
| rain-prob-night | Number:Dimensionless | % | Night-time is defined as those forecast times that fall between local dusk and dawn. |
| day-prob-heavy-rain | Number:Dimensionless | % | Heavy rain is defined as >1mm/hr. Daytime is defined as those forecast times that fall between local dawn and dusk. |
| night-prob-heavy-rain | Number:Dimensionless | % | Heavy rain is defined as >1mm/hr. Night-time is defined as those forecast times that fall between local dusk and dawn. |
| hail-prob-day | Number:Dimensionless | % | Daytime is defined as those forecast times that fall between local dawn and dusk. |
| hail-prob-night | Number:Dimensionless | % | Night-time is defined as those forecast times that fall between local dusk and dawn. |
| sferics-prob-day | Number:Dimensionless | % | This is the probability of a strike within a radius of 50km. |
| sferics-prob-night | Number:Dimensionless | % | This is the probability of a strike within a radius of 50km. |
#### Channel Groups for Daily Forecast Channels
| Channel Id | Description |
|-----------------------|---------------------------------------------------|
| daily-forecast | This is the weather forecast for the current day. |
| daily-forecast-plus01 | This is the weather forecast in 1 day. |
| daily-forecast-plus02 | This is the weather forecast in 2 days. |
| ..................... | ................................................. |
| daily-forecast-plus05 | This is the weather forecast in 5 days. |
| daily-forecast-plus06 | This is the weather forecast in 6 days. |
## Full Example
### Configuration (*.things)
```java
Bridge metofficedatahub:site:metoffice [siteRateDailyLimit=200, siteApiKey="<Site Specific API Key>"] {
site londonForecast "London Forecast" [hourlyForecastPollRate=1, dailyForecastPollRate=3, location="51.509865,-0.118092"]
}
```
### Configuration (*.items)
#### Hourly Forecast `example.items`
```java
Group gCurrentHourForecast "Current Hour Forecast"
Group gLondon "London"
Group gLondonCurrentHour "London Current Forecast" (gLondon,gCurrentHourForecast)
DateTime ForecastLondonHourlyForecastTs (gLondonCurrentHour) { channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#forecast-ts" }
Number:Temperature ForecastLondonCurrentHour (gLondonCurrentHour) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#air-temp-current" }
Number:Temperature ForecastLondonMinTemp (gLondonCurrentHour) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#air-temp-min" }
Number:Temperature ForecastLondonMaxTemp (gLondonCurrentHour) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#air-temp-max" }
Number:Temperature ForecastLondonFeelsLikeTemp (gLondonCurrentHour) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#feels-like" }
Number:Dimensionless ForecastLondonRelHumidity (gLondonCurrentHour) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#humidity" }
Number:Length ForecastLondonVisibility (gLondonCurrentHour) { unit="m",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#visibility" }
Number:Dimensionless ForecastLondonPrecipitationProb (gLondonCurrentHour) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#precip-prob" }
Number:Speed ForecastLondonPrecipitationRate (gLondonCurrentHour) { unit="mm/h",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#precip-rate" }
Number:Length ForecastLondonPrecipitationAmount (gLondonCurrentHour) { unit="mm",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#precip-total" }
Number:Length ForecastLondonSnowAmount (gLondonCurrentHour) { unit="mm",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#snow-total" }
Number:Dimensionless ForecastLondonUvIndex (gLondonCurrentHour) { channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#uv-index" }
Number:Pressure ForecastLondonpressure (gLondonCurrentHour) { unit="Pa",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#pressure" }
Number:Speed ForecastLondon10mWindSpeed (gLondonCurrentHour) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#wind-speed" }
Number:Speed ForecastLondon10mGustWindSpeed (gLondonCurrentHour) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#wind-speed-gust" }
Number:Speed ForecastLondon10mMaxGustWindSpeed (gLondonCurrentHour) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#wind-gust-max" }
Number:Angle ForecastLondon10mWindDirection (gLondonCurrentHour) { unit="°",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#wind-direction" }
Number:Temperature ForecastLondonDewPointTemp (gLondonCurrentHour) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast#dewpoint" }
Group gCurrentHourPlus01Forecast "Next Hours Forecast"
Group gLondonNextHour "London Next Hours Forecast" (gLondon,gCurrentHourPlus01Forecast)
DateTime ForecastLondonPlus01HourlyForecastTs (gLondonNextHour) { channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#forecast-ts" }
Number:Temperature ForecastLondonPlus01CurrentHour (gLondonNextHour) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#air-temp-current" }
Number:Temperature ForecastLondonPlus01MinTemp (gLondonNextHour) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#air-temp-min" }
Number:Temperature ForecastLondonPlus01MaxTemp (gLondonNextHour) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#air-temp-max" }
Number:Temperature ForecastLondonPlus01FeelsLikeTemp (gLondonNextHour) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#feels-like" }
Number:Dimensionless ForecastLondonPlus01RelHumidity (gLondonNextHour) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#humidity" }
Number:Length ForecastLondonPlus01Visibility (gLondonNextHour) { unit="m",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#visibility" }
Number:Speed ForecastLondonPlus01PrecipitationRate (gLondonNextHour) { unit="mm/h",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#precip-rate" }
Number:Dimensionless ForecastLondonPlus01PrecipitationProb (gLondonNextHour) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#precip-prob" }
Number:Length ForecastLondonPlus01PrecipitationAmount (gLondonNextHour) { unit="mm",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#precip-total" }
Number:Length ForecastLondonPlus01SnowAmount (gLondonNextHour) { unit="mm",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#snow-total" }
Number:Dimensionless ForecastLondonPlus01UvIndex (gLondonNextHour) { channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#uv-index" }
Number:Pressure ForecastLondonPlus01pressure (gLondonNextHour) { unit="Pa",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#pressure" }
Number:Speed ForecastLondonPlus0110mWindSpeed (gLondonNextHour) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#wind-speed" }
Number:Speed ForecastLondonPlus0110mGustWindSpeed (gLondonNextHour) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#wind-speed-gust" }
Number:Speed ForecastLondonPlus0110mMaxGustWindSpeed (gLondonNextHour) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#wind-gust-max" }
Number:Angle ForecastLondonPlus0110mWindDirection (gLondonNextHour) { unit="°",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#wind-direction" }
Number:Temperature ForecastLondonPlus01DewPointTemp (gLondonNextHour) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:current-forecast-plus01#dewpoint" }
```
#### Daily Forecast `example.items`
```java
Group gdaily-forecast "Current Daily Forecast"
Group gLondonCurrentDay "London Current Forecast" (gLondon,gdaily-forecast)
DateTime ForecastLondonDailyForecastTs (gLondonCurrentDay) { channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#forecast-ts" }
Number:Speed ForecastLondonMiddayWindSpeed10m (gLondonCurrentDay) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#wind-speed-day" }
Number:Speed ForecastLondonMidnightWindSpeed10m (gLondonCurrentDay) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#wind-speed-night" }
Number:Angle ForecastLondonMidday10MWindDirection (gLondonCurrentDay) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#wind-direction-day" }
Number:Angle ForecastLondonMidnight10MWindDirection (gLondonCurrentDay) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#wind-direction-night" }
Number:Speed ForecastLondonMidday10mWindGust (gLondonCurrentDay) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#wind-gust-day" }
Number:Speed ForecastLondonMidnight10mWindGust (gLondonCurrentDay) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#wind-gust-night" }
Number:Length ForecastLondonMiddayVisibility (gLondonCurrentDay) { unit="m",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#visibility-day" }
Number:Length ForecastLondonMidnightVisibility (gLondonCurrentDay) { unit="m",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#visibility-night" }
Number:Dimensionless ForecastLondonMiddayRelativeHumidity (gLondonCurrentDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#humidity-day" }
Number:Dimensionless ForecastLondonMidnightRelativeHumidity (gLondonCurrentDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#humidity-night" }
Number:Pressure ForecastLondonMiddaypressure (gLondonCurrentDay) { unit="Pa",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#pressure-day" }
Number:Pressure ForecastLondonMidnightpressure (gLondonCurrentDay) { unit="Pa",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#pressure-night" }
Number:Dimensionless ForecastLondonMaxUvIndex (gLondonCurrentDay) { channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#uv-max" }
Number:Temperature ForecastLondonNightUpperBoundMinTemp (gLondonCurrentDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#temp-min-ub-night" }
Number:Temperature ForecastLondonDayLowerBoundMaxTemp (gLondonCurrentDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#temp-max-lb-day" }
Number:Temperature ForecastLondonNightLowerBoundMinTemp (gLondonCurrentDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#temp-min-lb-night" }
Number:Temperature ForecastLondonDayMaxFeelsLikeTemp (gLondonCurrentDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#feels-like-max-day" }
Number:Temperature ForecastLondonNightMinFeelsLikeTemp (gLondonCurrentDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#feels-like-min-night" }
Number:Temperature ForecastLondonDayMaxScreenTemperature (gLondonCurrentDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#temp-max-day" }
Number:Temperature ForecastLondonNightMinScreenTemperature (gLondonCurrentDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#temp-min-night" }
Number:Temperature ForecastLondonDayUpperBoundMaxFeelsLikeTemp (gLondonCurrentDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#feels-like-max-ub-day" }
Number:Temperature ForecastLondonNightUpperBoundMinFeelsLikeTemp (gLondonCurrentDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#feels-like-min-ub-night" }
Number:Temperature ForecastLondonDayLowerBoundMaxFeelsLikeTemp (gLondonCurrentDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#feels-like-max-lb-day" }
Number:Temperature ForecastLondonNightLowerBoundMinFeelsLikeTemp (gLondonCurrentDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#feels-like-min-lb-night" }
Number:Dimensionless ForecastLondonDayProbabilityOfPrecipitation (gLondonCurrentDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#precip-prob-day" }
Number:Dimensionless ForecastLondonNightProbabilityOfPrecipitation (gLondonCurrentDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#precip-prob-night" }
Number:Dimensionless ForecastLondonDayProbabilityOfSnow (gLondonCurrentDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#snow-prob-day" }
Number:Dimensionless ForecastLondonNightProbabilityOfSnow (gLondonCurrentDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#snow-prob-night" }
Number:Dimensionless ForecastLondonDayProbabilityOfHeavySnow (gLondonCurrentDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#heavy-snow-prob-day" }
Number:Dimensionless ForecastLondonNightProbabilityOfHeavySnow (gLondonCurrentDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#heavy-snow-prob-night" }
Number:Dimensionless ForecastLondonDayProbabilityOfRain (gLondonCurrentDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#rain-prob-day" }
Number:Dimensionless ForecastLondonNightProbabilityOfRain (gLondonCurrentDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#rain-prob-night" }
Number:Dimensionless ForecastLondonDayProbabilityOfHeavyRain (gLondonCurrentDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#day-prob-heavy-rain" }
Number:Dimensionless ForecastLondonNightProbabilityOfHeavyRain (gLondonCurrentDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#night-prob-heavy-rain" }
Number:Dimensionless ForecastLondonDayProbabilityOfHail (gLondonCurrentDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#hail-prob-day" }
Number:Dimensionless ForecastLondonNightProbabilityOfHail (gLondonCurrentDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#hail-prob-night" }
Number:Dimensionless ForecastLondonDayProbabilityOfSferics (gLondonCurrentDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#sferics-prob-day" }
Number:Dimensionless ForecastLondonNightProbabilityOfSferics (gLondonCurrentDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast#sferics-prob-night" }
Group gCurrentDailyPlus01Forecast "Current Day +1 Daily Forecast"
Group gLondonNextDay "London Next Day Forecast" (gLondon,gCurrentDailyPlus01Forecast)
DateTime ForecastLondonPlus01DailyForecastTs (gLondonNextDay) { channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#forecast-ts" }
Number:Speed ForecastLondonPlus01MiddayWindSpeed10m (gLondonNextDay) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#wind-speed-day" }
Number:Speed ForecastLondonPlus01MidnightWindSpeed10m (gLondonNextDay) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#wind-speed-night" }
Number:Angle ForecastLondonPlus01Midday10MWindDirection (gLondonNextDay) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#wind-direction-day" }
Number:Angle ForecastLondonPlus01Midnight10MWindDirection (gLondonNextDay) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#wind-direction-night" }
Number:Speed ForecastLondonPlus01Midday10mWindGust (gLondonNextDay) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#wind-gust-day" }
Number:Speed ForecastLondonPlus01Midnight10mWindGust (gLondonNextDay) { unit="m/s",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#wind-gust-night" }
Number:Length ForecastLondonPlus01MiddayVisibility (gLondonNextDay) { unit="m",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#visibility-day" }
Number:Length ForecastLondonPlus01MidnightVisibility (gLondonNextDay) { unit="m",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#visibility-night" }
Number:Dimensionless ForecastLondonPlus01MiddayRelativeHumidity (gLondonNextDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#humidity-day" }
Number:Dimensionless ForecastLondonPlus01MidnightRelativeHumidity (gLondonNextDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#humidity-night" }
Number:Pressure ForecastLondonPlus01Middaypressure (gLondonNextDay) { unit="Pa",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#pressure-day" }
Number:Pressure ForecastLondonPlus01Midnightpressure (gLondonNextDay) { unit="Pa",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#pressure-night" }
Number:Dimensionless ForecastLondonPlus01MaxUvIndex (gLondonNextDay) { channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#uv-max" }
Number:Temperature ForecastLondonPlus01NightUpperBoundMinTemp (gLondonNextDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#temp-min-ub-night" }
Number:Temperature ForecastLondonPlus01DayLowerBoundMaxTemp (gLondonNextDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#temp-max-lb-day" }
Number:Temperature ForecastLondonPlus01NightLowerBoundMinTemp (gLondonNextDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#temp-min-lb-night" }
Number:Temperature ForecastLondonPlus01DayMaxFeelsLikeTemp (gLondonNextDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#feels-like-max-day" }
Number:Temperature ForecastLondonPlus01NightMinFeelsLikeTemp (gLondonNextDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#feels-like-min-night" }
Number:Temperature ForecastLondonPlus01DayMaxScreenTemperature (gLondonNextDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#temp-max-day" }
Number:Temperature ForecastLondonPlus01NightMinScreenTemperature (gLondonNextDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#temp-min-night" }
Number:Temperature ForecastLondonPlus01DayUpperBoundMaxFeelsLikeTemp (gLondonNextDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#feels-like-max-ub-day" }
Number:Temperature ForecastLondonPlus01NightUpperBoundMinFeelsLikeTemp (gLondonNextDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#feels-like-min-ub-night" }
Number:Temperature ForecastLondonPlus01DayLowerBoundMaxFeelsLikeTemp (gLondonNextDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#feels-like-max-lb-day" }
Number:Temperature ForecastLondonPlus01NightLowerBoundMinFeelsLikeTemp (gLondonNextDay) { unit="°C",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#feels-like-min-lb-night" }
Number:Dimensionless ForecastLondonPlus01DayProbabilityOfPrecipitation (gLondonNextDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#precip-prob-day" }
Number:Dimensionless ForecastLondonPlus01NightProbabilityOfPrecipitation (gLondonNextDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#precip-prob-night" }
Number:Dimensionless ForecastLondonPlus01DayProbabilityOfSnow (gLondonNextDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#snow-prob-day" }
Number:Dimensionless ForecastLondonPlus01NightProbabilityOfSnow (gLondonNextDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#snow-prob-night" }
Number:Dimensionless ForecastLondonPlus01DayProbabilityOfHeavySnow (gLondonNextDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#heavy-snow-prob-day" }
Number:Dimensionless ForecastLondonPlus01NightProbabilityOfHeavySnow (gLondonNextDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#heavy-snow-prob-night" }
Number:Dimensionless ForecastLondonPlus01DayProbabilityOfRain (gLondonNextDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#rain-prob-day" }
Number:Dimensionless ForecastLondonPlus01NightProbabilityOfRain (gLondonNextDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#rain-prob-night" }
Number:Dimensionless ForecastLondonPlus01DayProbabilityOfHeavyRain (gLondonNextDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#day-prob-heavy-rain" }
Number:Dimensionless ForecastLondonPlus01NightProbabilityOfHeavyRain (gLondonNextDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#night-prob-heavy-rain" }
Number:Dimensionless ForecastLondonPlus01DayProbabilityOfHail (gLondonNextDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#hail-prob-day" }
Number:Dimensionless ForecastLondonPlus01NightProbabilityOfHail (gLondonNextDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#hail-prob-night" }
Number:Dimensionless ForecastLondonPlus01DayProbabilityOfSferics (gLondonNextDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#sferics-prob-day" }
Number:Dimensionless ForecastLondonPlus01NightProbabilityOfSferics (gLondonNextDay) { unit="%",channel="metofficedatahub:site:metoffice:londonForecast:daily-forecast-plus01#sferics-prob-night" }
```
### Configuration (*.sitemap)
#### Hourly Forecast `example.sitemap`
```perl
Frame {
Text item=ForecastLondonHourlyForecastTs icon="time"
Text item=ForecastLondonCurrentHour icon="temperature"
Text item=ForecastLondonMinTemp icon="temperature"
Text item=ForecastLondonMaxTemp icon="temperature"
Text item=ForecastLondonFeelsLikeTemp icon="temperature"
Text item=ForecastLondonRelHumidity icon="humidity"
Text item=ForecastLondonVisibility icon="sun_clouds"
Text item=ForecastLondonPrecipitationRate icon="rain"
Text item=ForecastLondonPrecipitationProb icon="rain"
Text item=ForecastLondonPrecipitationAmount icon="rain"
Text item=ForecastLondonSnowAmount icon="rain"
Text item=ForecastLondonUvIndex icon="sun"
Text item=ForecastLondonpressure icon="pressure"
Text item=ForecastLondon10mWindSpeed icon="wind"
Text item=ForecastLondon10mGustWindSpeed icon="wind"
Text item=ForecastLondon10mMaxGustWindSpeed icon="wind"
Text item=ForecastLondon10mWindDirection icon="wind"
Text item=ForecastLondonDewPointTemp icon="temperature"
}
Frame {
Text item=ForecastLondonPlus01HourlyForecastTs icon="time"
Text item=ForecastLondonPlus01CurrentHour icon="temperature"
Text item=ForecastLondonPlus01MinTemp icon="temperature"
Text item=ForecastLondonPlus01MaxTemp icon="temperature"
Text item=ForecastLondonPlus01FeelsLikeTemp icon="temperature"
Text item=ForecastLondonPlus01RelHumidity icon="humidity"
Text item=ForecastLondonPlus01Visibility icon="sun_clouds"
Text item=ForecastLondonPlus01PrecipitationRate icon="rain"
Text item=ForecastLondonPlus01PrecipitationProb icon="rain"
Text item=ForecastLondonPlus01PrecipitationAmount icon="rain"
Text item=ForecastLondonPlus01SnowAmount icon="rain"
Text item=ForecastLondonPlus01UvIndex icon="sun"
Text item=ForecastLondonPlus01pressure icon="pressure"
Text item=ForecastLondonPlus0110mWindSpeed icon="wind"
Text item=ForecastLondonPlus0110mGustWindSpeed icon="wind"
Text item=ForecastLondonPlus0110mMaxGustWindSpeed icon="wind"
Text item=ForecastLondonPlus0110mWindDirection icon="wind"
Text item=ForecastLondonPlus01DewPointTemp icon="temperature"
}
```
#### Daily Forecast `example.items`
```perl
Frame {
Text item=ForecastLondonDailyForecastTs icon="time"
Text item=ForecastLondonMiddayWindSpeed10m icon="wind"
Text item=ForecastLondonMidnightWindSpeed10m icon="wind"
Text item=ForecastLondonMidday10MWindDirection icon="wind"
Text item=ForecastLondonMidnight10MWindDirection icon="wind"
Text item=ForecastLondonMidday10mWindGust icon="wind"
Text item=ForecastLondonMidnight10mWindGust icon="wind"
Text item=ForecastLondonMiddayVisibility icon="sun_clouds"
Text item=ForecastLondonMidnightVisibility icon="sun_clouds"
Text item=ForecastLondonMiddayRelativeHumidity icon="humidity"
Text item=ForecastLondonMidnightRelativeHumidity icon="humidity"
Text item=ForecastLondonMiddaypressure icon="pressure"
Text item=ForecastLondonMidnightpressure icon="pressure"
Text item=ForecastLondonMaxUvIndex icon="pressure"
Text item=ForecastLondonNightUpperBoundMinTemp icon="temperature"
Text item=ForecastLondonDayLowerBoundMaxTemp icon="temperature"
Text item=ForecastLondonNightLowerBoundMinTemp icon="temperature"
Text item=ForecastLondonDayMaxFeelsLikeTemp icon="temperature"
Text item=ForecastLondonNightMinFeelsLikeTemp icon="temperature"
Text item=ForecastLondonDayMaxScreenTemperature icon="temperature"
Text item=ForecastLondonNightMinScreenTemperature icon="temperature"
Text item=ForecastLondonDayUpperBoundMaxFeelsLikeTemp icon="temperature"
Text item=ForecastLondonNightUpperBoundMinFeelsLikeTemp icon="temperature"
Text item=ForecastLondonDayLowerBoundMaxFeelsLikeTemp icon="temperature"
Text item=ForecastLondonNightLowerBoundMinFeelsLikeTemp icon="temperature"
Text item=ForecastLondonDayProbabilityOfPrecipitation icon="rain"
Text item=ForecastLondonNightProbabilityOfPrecipitation icon="rain"
Text item=ForecastLondonDayProbabilityOfSnow icon="rain"
Text item=ForecastLondonNightProbabilityOfSnow icon="rain"
Text item=ForecastLondonDayProbabilityOfHeavySnow icon="rain"
Text item=ForecastLondonNightProbabilityOfHeavySnow icon="rain"
Text item=ForecastLondonDayProbabilityOfRain icon="rain"
Text item=ForecastLondonNightProbabilityOfRain icon="rain"
Text item=ForecastLondonDayProbabilityOfHeavyRain icon="rain"
Text item=ForecastLondonNightProbabilityOfHeavyRain icon="rain"
Text item=ForecastLondonDayProbabilityOfHail icon="rain"
Text item=ForecastLondonNightProbabilityOfHail icon="rain"
Text item=ForecastLondonDayProbabilityOfSferics icon="line"
Text item=ForecastLondonNightProbabilityOfSferics icon="line"
}
Frame {
Text item=ForecastLondonPlus01DailyForecastTs icon="time"
Text item=ForecastLondonPlus01MiddayWindSpeed10m icon="wind"
Text item=ForecastLondonPlus01MidnightWindSpeed10m icon="wind"
Text item=ForecastLondonPlus01Midday10MWindDirection icon="wind"
Text item=ForecastLondonPlus01Midnight10MWindDirection icon="wind"
Text item=ForecastLondonPlus01Midday10mWindGust icon="wind"
Text item=ForecastLondonPlus01Midnight10mWindGust icon="wind"
Text item=ForecastLondonPlus01MiddayVisibility icon="sun_clouds"
Text item=ForecastLondonPlus01MidnightVisibility icon="sun_clouds"
Text item=ForecastLondonPlus01MiddayRelativeHumidity icon="humidity"
Text item=ForecastLondonPlus01MidnightRelativeHumidity icon="humidity"
Text item=ForecastLondonPlus01Middaypressure icon="pressure"
Text item=ForecastLondonPlus01Midnightpressure icon="pressure"
Text item=ForecastLondonPlus01MaxUvIndex icon="pressure"
Text item=ForecastLondonPlus01NightUpperBoundMinTemp icon="temperature"
Text item=ForecastLondonPlus01DayLowerBoundMaxTemp icon="temperature"
Text item=ForecastLondonPlus01NightLowerBoundMinTemp icon="temperature"
Text item=ForecastLondonPlus01DayMaxFeelsLikeTemp icon="temperature"
Text item=ForecastLondonPlus01NightMinFeelsLikeTemp icon="temperature"
Text item=ForecastLondonPlus01DayMaxScreenTemperature icon="temperature"
Text item=ForecastLondonPlus01NightMinScreenTemperature icon="temperature"
Text item=ForecastLondonPlus01DayUpperBoundMaxFeelsLikeTemp icon="temperature"
Text item=ForecastLondonPlus01NightUpperBoundMinFeelsLikeTemp icon="temperature"
Text item=ForecastLondonPlus01DayLowerBoundMaxFeelsLikeTemp icon="temperature"
Text item=ForecastLondonPlus01NightLowerBoundMinFeelsLikeTemp icon="temperature"
Text item=ForecastLondonPlus01DayProbabilityOfPrecipitation icon="rain"
Text item=ForecastLondonPlus01NightProbabilityOfPrecipitation icon="rain"
Text item=ForecastLondonPlus01DayProbabilityOfSnow icon="rain"
Text item=ForecastLondonPlus01NightProbabilityOfSnow icon="rain"
Text item=ForecastLondonPlus01DayProbabilityOfHeavySnow icon="rain"
Text item=ForecastLondonPlus01NightProbabilityOfHeavySnow icon="rain"
Text item=ForecastLondonPlus01DayProbabilityOfRain icon="rain"
Text item=ForecastLondonPlus01NightProbabilityOfRain icon="rain"
Text item=ForecastLondonPlus01DayProbabilityOfHeavyRain icon="rain"
Text item=ForecastLondonPlus01NightProbabilityOfHeavyRain icon="rain"
Text item=ForecastLondonPlus01DayProbabilityOfHail icon="rain"
Text item=ForecastLondonPlus01NightProbabilityOfHail icon="rain"
Text item=ForecastLondonPlus01DayProbabilityOfSferics icon="line"
Text item=ForecastLondonPlus01NightProbabilityOfSferics icon="line"
}
```

View File

@ -0,0 +1,17 @@
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>org.openhab.addons.bundles</groupId>
<artifactId>org.openhab.addons.reactor.bundles</artifactId>
<version>4.3.0-SNAPSHOT</version>
</parent>
<artifactId>org.openhab.binding.metofficedatahub</artifactId>
<name>openHAB Add-ons :: Bundles :: MetOffice DataHub Binding</name>
</project>

View File

@ -0,0 +1,9 @@
<?xml version="1.0" encoding="UTF-8"?>
<features name="org.openhab.binding.metofficedatahub-${project.version}" xmlns="http://karaf.apache.org/xmlns/features/v1.4.0">
<repository>mvn:org.openhab.core.features.karaf/org.openhab.core.features.karaf.openhab-core/${ohc.version}/xml/features</repository>
<feature name="openhab-binding-metofficedatahub" description="MetOfficeDataHub Binding" version="${project.version}">
<feature>openhab-runtime-base</feature>
<bundle start-level="80">mvn:org.openhab.addons.bundles/org.openhab.binding.metofficedatahub/${project.version}</bundle>
</feature>
</features>

View File

@ -0,0 +1,183 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal;
import java.util.Random;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.openhab.core.thing.ThingTypeUID;
import com.google.gson.FieldNamingPolicy;
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
/**
* The {@link MetOfficeDataHubBindingConstants} class defines common constants, which are
* used across the whole binding.
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public class MetOfficeDataHubBindingConstants {
public static final Gson GSON = new GsonBuilder()
.setFieldNamingPolicy(FieldNamingPolicy.LOWER_CASE_WITH_UNDERSCORES).setPrettyPrinting()
.disableHtmlEscaping().serializeNulls().create();
private static final String BINDING_ID = "metofficedatahub";
// List of all Thing Type UIDs
public static final ThingTypeUID THING_TYPE_BRIDGE = new ThingTypeUID(BINDING_ID, "account");
public static final ThingTypeUID THING_TYPE_SITE_SPEC_API = new ThingTypeUID(BINDING_ID, "site");
/**
* Site Specific API - Shared
*/
public static final String SITE_TIMESTAMP = "forecast-ts";
/**
* Site Specific API - Hourly Forecast Channel Names
*/
public static final String SITE_HOURLY_FORECAST_SCREEN_TEMPERATURE = "air-temp-current";
public static final String SITE_HOURLY_FORECAST_MIN_SCREEN_TEMPERATURE = "air-temp-min";
public static final String SITE_HOURLY_FORECAST_MAX_SCREEN_TEMPERATURE = "air-temp-max";
public static final String SITE_HOURLY_FEELS_LIKE_TEMPERATURE = "feels-like";
public static final String SITE_HOURLY_SCREEN_RELATIVE_HUMIDITY = "humidity";
public static final String SITE_HOURLY_VISIBILITY = "visibility";
public static final String SITE_HOURLY_PROBABILITY_OF_PRECIPITATION = "precip-prob";
public static final String SITE_HOURLY_PRECIPITATION_RATE = "precip-rate";
public static final String SITE_HOURLY_TOTAL_PRECIPITATION_AMOUNT = "precip-total";
public static final String SITE_HOURLY_TOTAL_SNOW_AMOUNT = "snow-total";
public static final String SITE_HOURLY_UV_INDEX = "uv-index";
public static final String SITE_HOURLY_PRESSURE = "pressure";
public static final String SITE_HOURLY_WIND_SPEED_10M = "wind-speed";
public static final String SITE_HOURLY_WIND_GUST_SPEED_10M = "wind-speed-gust";
public static final String SITE_HOURLY_MAX_10M_WIND_GUST = "wind-gust-max";
public static final String SITE_HOURLY_WIND_DIRECTION_FROM_10M = "wind-direction";
public static final String SITE_HOURLY_SCREEN_DEW_POINT_TEMPERATURE = "dewpoint";
public static final String SITE_DAILY_MIDDAY_WIND_SPEED_10M = "wind-speed-day";
public static final String SITE_DAILY_MIDNIGHT_WIND_SPEED_10M = "wind-speed-night";
public static final String SITE_DAILY_MIDDAY_WIND_DIRECTION_10M = "wind-direction-day";
public static final String SITE_DAILY_MIDNIGHT_WIND_DIRECTION_10M = "wind-direction-night";
public static final String SITE_DAILY_MIDDAY_WIND_GUST_10M = "wind-gust-day";
public static final String SITE_DAILY_MIDNIGHT_WIND_GUST_10M = "wind-gust-night";
public static final String SITE_DAILY_MIDDAY_VISIBILITY = "visibility-day";
public static final String SITE_DAILY_MIDNIGHT_VISIBILITY = "visibility-night";
public static final String SITE_DAILY_MIDDAY_REL_HUMIDITY = "humidity-day";
public static final String SITE_DAILY_MIDNIGHT_REL_HUMIDITY = "humidity-night";
public static final String SITE_DAILY_MIDDAY_PRESSURE = "pressure-day";
public static final String SITE_DAILY_MIDNIGHT_PRESSURE = "pressure-night";
public static final String SITE_DAILY_DAY_MAX_UV_INDEX = "uv-max";
public static final String SITE_DAILY_DAY_UPPER_BOUND_MAX_TEMP = "temp-max-ub-day";
public static final String SITE_DAILY_DAY_LOWER_BOUND_MAX_TEMP = "temp-max-lb-day";
public static final String SITE_DAILY_NIGHT_UPPER_BOUND_MAX_TEMP = "temp-min-ub-night";
public static final String SITE_DAILY_NIGHT_LOWER_BOUND_MAX_TEMP = "temp-min-lb-night";
public static final String SITE_DAILY_NIGHT_FEELS_LIKE_MIN_TEMP = "feels-like-min-night";
public static final String SITE_DAILY_DAY_FEELS_LIKE_MAX_TEMP = "feels-like-max-day";
public static final String SITE_DAILY_NIGHT_LOWER_BOUND_MIN_TEMP = "temp-min-lb-night";
public static final String SITE_DAILY_DAY_MAX_FEELS_LIKE_TEMP = "feels-like-max-day";
public static final String SITE_DAILY_NIGHT_LOWER_BOUND_MIN_FEELS_LIKE_TEMP = "feels-like-min-lb-night";
public static final String SITE_DAILY_DAY_LOWER_BOUND_MAX_FEELS_LIKE_TEMP = "feels-like-max-lb-day";
public static final String SITE_DAILY_DAY_UPPER_BOUND_MAX_FEELS_LIKE_TEMP = "feels-like-max-ub-day";
public static final String SITE_DAILY_UPPER_BOUND_MIN_FEELS_LIKE_TEMP = "feels-like-min-ub-night";
public static final String SITE_DAILY_DAY_PROBABILITY_OF_PRECIPITATION = "precip-prob-day";
public static final String SITE_DAILY_NIGHT_PROBABILITY_OF_PRECIPITATION = "precip-prob-night";
public static final String SITE_DAILY_DAY_PROBABILITY_OF_SNOW = "snow-prob-day";
public static final String SITE_DAILY_NIGHT_PROBABILITY_OF_SNOW = "snow-prob-night";
public static final String SITE_DAILY_DAY_PROBABILITY_OF_HEAVY_SNOW = "heavy-snow-prob-day";
public static final String SITE_DAILY_NIGHT_PROBABILITY_OF_HEAVY_SNOW = "heavy-snow-prob-night";
public static final String SITE_DAILY_DAY_PROBABILITY_OF_RAIN = "rain-prob-day";
public static final String SITE_DAILY_NIGHT_PROBABILITY_OF_RAIN = "rain-prob-night";
public static final String SITE_DAILY_DAY_PROBABILITY_OF_HEAVY_RAIN = "day-prob-heavy-rain";
public static final String SITE_DAILY_NIGHT_PROBABILITY_OF_HEAVY_RAIN = "night-prob-heavy-rain";
public static final String SITE_DAILY_DAY_PROBABILITY_OF_HAIL = "hail-prob-day";
public static final String SITE_DAILY_NIGHT_PROBABILITY_OF_HAIL = "hail-prob-night";
public static final String SITE_DAILY_DAY_PROBABILITY_OF_SFERICS = "sferics-prob-day";
public static final String SITE_DAILY_NIGHT_PROBABILITY_OF_SFERICS = "sferics-prob-night";
public static final String SITE_DAILY_DAY_MAX_SCREEN_TEMPERATURE = "temp-max-day";
public static final String SITE_DAILY_NIGHT_MIN_SCREEN_TEMPERATURE = "temp-min-night";
public static final String GROUP_PREFIX_HOURS_FORECAST = "current-forecast";
public static final String GROUP_PREFIX_DAILY_FORECAST = "daily-forecast";
public static final String GROUP_POSTFIX_BOTH_FORECASTS = "-plus";
public static final char GROUP_PREFIX_TO_ITEM = '#';
public static final String GET_FORECAST_URL_DAILY = "https://data.hub.api.metoffice.gov.uk/sitespecific/v0/point/daily?latitude=<LATITUDE>&longitude=<LONGITUDE>";
public static final String GET_FORECAST_URL_HOURLY = "https://data.hub.api.metoffice.gov.uk/sitespecific/v0/point/hourly?latitude=<LATITUDE>&longitude=<LONGITUDE>";
public static final String GET_FORECAST_KEY_LATITUDE = "<LATITUDE>";
public static final String GET_FORECAST_KEY_LONGITUDE = "<LONGITUDE>";
public static final String GET_FORECAST_API_KEY_HEADER = "apikey";
public static final int GET_FORECAST_REQUEST_TIMEOUT_SECONDS = 3;
public static final String EXPECTED_TS_FORMAT = "YYYY-MM-dd HH:mm:ss.SSS";
public static final long DAY_IN_MILLIS = 86400000;
public static final Random RANDOM_GENERATOR = new Random();
public static final String BRIDGE_PROP_FORECAST_REQUEST_COUNT = "Site Specific API Call Count";
public static final Runnable NO_OP = () -> {
};
}

View File

@ -0,0 +1,35 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal;
import org.eclipse.jdt.annotation.NonNullByDefault;
/**
* The {@link MetOfficeDataHubBridgeConfiguration} class contains fields mapping thing configuration parameters.
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public class MetOfficeDataHubBridgeConfiguration {
/**
* Site Specific API Subscription - API Key
*/
public String siteApiKey = "";
/**
* Rate limit of API call's in 24 hour period starting from 0000 (Free is capped at 360 - this allows 110 due to
* reboots)
*/
public int siteRateDailyLimit = 250;
}

View File

@ -0,0 +1,159 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal;
import static org.openhab.binding.metofficedatahub.internal.MetOfficeDataHubBindingConstants.BRIDGE_PROP_FORECAST_REQUEST_COUNT;
import java.util.HashMap;
import java.util.Map;
import java.util.Objects;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
import org.openhab.binding.metofficedatahub.internal.api.IConnectionStatusListener;
import org.openhab.binding.metofficedatahub.internal.api.IRateLimiterListener;
import org.openhab.binding.metofficedatahub.internal.api.RequestLimiter;
import org.openhab.binding.metofficedatahub.internal.api.SiteApi;
import org.openhab.core.i18n.LocaleProvider;
import org.openhab.core.i18n.TimeZoneProvider;
import org.openhab.core.i18n.TranslationProvider;
import org.openhab.core.io.net.http.HttpClientFactory;
import org.openhab.core.storage.StorageService;
import org.openhab.core.thing.Bridge;
import org.openhab.core.thing.ChannelUID;
import org.openhab.core.thing.Thing;
import org.openhab.core.thing.ThingStatus;
import org.openhab.core.thing.ThingStatusDetail;
import org.openhab.core.thing.binding.BaseBridgeHandler;
import org.openhab.core.types.Command;
import org.osgi.framework.Bundle;
import org.osgi.framework.FrameworkUtil;
import org.osgi.service.component.annotations.Reference;
/**
* The {@link MetOfficeDataHubBridgeHandler} models the account(s) to the MetOfficeDataHub services.
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public class MetOfficeDataHubBridgeHandler extends BaseBridgeHandler
implements IRateLimiterListener, IConnectionStatusListener {
private volatile MetOfficeDataHubBridgeConfiguration config = getConfigAs(
MetOfficeDataHubBridgeConfiguration.class);
private final TranslationProvider translationProvider;
private final LocaleProvider localeProvider;
private final Bundle bundle;
private SiteApi siteApi;
private String bridgeId = "";
public MetOfficeDataHubBridgeHandler(final Bridge bridge, @Reference HttpClientFactory httpClientFactory,
@Reference TranslationProvider translationProvider, @Reference LocaleProvider localeProvider,
@Reference StorageService storageService, @Reference TimeZoneProvider timeZoneProvider) {
super(bridge);
bridgeId = getThing().getUID().getAsString();
this.translationProvider = translationProvider;
this.localeProvider = localeProvider;
this.bundle = FrameworkUtil.getBundle(getClass());
this.siteApi = new SiteApi(bridgeId, httpClientFactory, storageService, translationProvider, localeProvider,
timeZoneProvider, scheduler);
}
@Override
public void initialize() {
updateStatus(ThingStatus.UNKNOWN);
siteApi.registerListeners(bridgeId, this);
config = getConfigAs(MetOfficeDataHubBridgeConfiguration.class);
siteApi.setLimits(config.siteRateDailyLimit);
siteApi.setApiKey(config.siteApiKey);
siteApi.validateSiteApi();
}
@Override
public void dispose() {
siteApi.deregisterListeners(bridgeId, this);
siteApi.dispose();
}
@Override
public void handleCommand(ChannelUID channelUID, Command command) {
}
// API Management
public SiteApi getSiteApi() {
return siteApi;
}
// Localization functionality
public String getLocalizedText(String key, @Nullable Object @Nullable... arguments) {
String result = translationProvider.getText(bundle, key, key, localeProvider.getLocale(), arguments);
return Objects.nonNull(result) ? result : key;
}
// Implementation of IRateLimiterListener
@Override
public void processRateLimiterUpdated(RequestLimiter requestLimiter) {
final Map<String, String> newProps = new HashMap<>();
newProps.put(BRIDGE_PROP_FORECAST_REQUEST_COUNT, String.valueOf(requestLimiter.getCurrentRequestCount()));
this.updateProperties(newProps);
}
// Implementation of IConnectionStatusListener
@Override
public void processAuthenticationResult(boolean authenticated) {
if (!authenticated) {
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
getLocalizedText("bridge.error.site-specific.auth-issue"));
} else {
processConnected();
}
}
@Override
public void processCommunicationFailure(final @Nullable Throwable e) {
String message = "";
if (e != null) {
if (e.getLocalizedMessage() != null) {
message = e.getLocalizedMessage();
} else if (e.getMessage() != null) {
message = e.getMessage();
}
}
if (message == null || message.isBlank()) {
message = getLocalizedText("bridge.error.site-specific.communication-failure.unknown");
}
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR,
getLocalizedText("bridge.error.site-specific.communication-failure", message));
}
@Override
public void processConnected() {
updateStatus(ThingStatus.ONLINE);
for (Thing thing : getThing().getThings()) {
if (thing instanceof MetOfficeDataHubSiteHandler siteHandler) {
if (!siteHandler.requiresPoll()) {
siteHandler.scheduleInitTask();
}
}
}
}
}

View File

@ -0,0 +1,89 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal;
import static org.openhab.binding.metofficedatahub.internal.MetOfficeDataHubBindingConstants.*;
import java.util.Set;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
import org.openhab.core.i18n.LocaleProvider;
import org.openhab.core.i18n.LocationProvider;
import org.openhab.core.i18n.TimeZoneProvider;
import org.openhab.core.i18n.TranslationProvider;
import org.openhab.core.io.net.http.HttpClientFactory;
import org.openhab.core.storage.StorageService;
import org.openhab.core.thing.Bridge;
import org.openhab.core.thing.Thing;
import org.openhab.core.thing.ThingTypeUID;
import org.openhab.core.thing.binding.BaseThingHandlerFactory;
import org.openhab.core.thing.binding.ThingHandler;
import org.openhab.core.thing.binding.ThingHandlerFactory;
import org.osgi.service.component.annotations.Activate;
import org.osgi.service.component.annotations.Component;
import org.osgi.service.component.annotations.Reference;
/**
* The {@link MetOfficeDataHubHandlerFactory} is responsible for creating things and thing
* handlers.
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
@Component(configurationPid = "binding.metofficedatahub", service = ThingHandlerFactory.class)
public class MetOfficeDataHubHandlerFactory extends BaseThingHandlerFactory {
private static final Set<ThingTypeUID> SUPPORTED_THING_TYPES_UIDS = Set.of(THING_TYPE_BRIDGE,
THING_TYPE_SITE_SPEC_API);
private final LocationProvider locationProvider;
private final HttpClientFactory httpClientFactory;
private final TranslationProvider translationProvider;
private final LocaleProvider localeProvider;
private final TimeZoneProvider timeZoneProvider;
private final StorageService storageService;
@Activate
public MetOfficeDataHubHandlerFactory(@Reference LocationProvider locationProvider,
@Reference HttpClientFactory httpClientFactory, @Reference TranslationProvider translationProvider,
@Reference LocaleProvider localeProvider, @Reference TimeZoneProvider timeZoneProvider,
@Reference StorageService storageService) {
this.locationProvider = locationProvider;
this.httpClientFactory = httpClientFactory;
this.translationProvider = translationProvider;
this.localeProvider = localeProvider;
this.timeZoneProvider = timeZoneProvider;
this.storageService = storageService;
}
@Override
public boolean supportsThingType(ThingTypeUID thingTypeUID) {
return SUPPORTED_THING_TYPES_UIDS.contains(thingTypeUID);
}
@Override
protected @Nullable ThingHandler createHandler(Thing thing) {
ThingTypeUID thingTypeUID = thing.getThingTypeUID();
if (THING_TYPE_BRIDGE.equals(thingTypeUID)) {
return new MetOfficeDataHubBridgeHandler((Bridge) thing, httpClientFactory, translationProvider,
localeProvider, storageService, timeZoneProvider);
} else if (THING_TYPE_SITE_SPEC_API.equals(thingTypeUID)) {
return new MetOfficeDataHubSiteHandler(thing, locationProvider, translationProvider, localeProvider,
timeZoneProvider);
}
return null;
}
}

View File

@ -0,0 +1,30 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal;
import org.eclipse.jdt.annotation.NonNullByDefault;
/**
* The {@link MetOfficeDataHubSiteConfiguration} class contains fields mapping thing configuration parameters.
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public class MetOfficeDataHubSiteConfiguration {
public String location = "";
public int hourlyForecastPollRate = 1;
public int dailyForecastPollRate = 3;
}

View File

@ -0,0 +1,577 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal;
import static org.openhab.binding.metofficedatahub.internal.MetOfficeDataHubBindingConstants.*;
import static org.openhab.core.library.unit.MetricPrefix.MILLI;
import static org.openhab.core.library.unit.SIUnits.METRE;
import java.time.Duration;
import java.time.Instant;
import java.time.temporal.ChronoUnit;
import java.util.List;
import java.util.Objects;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;
import javax.measure.Unit;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
import org.openhab.binding.metofficedatahub.internal.api.ISiteResponseListener;
import org.openhab.binding.metofficedatahub.internal.api.ResponseDataProcessor;
import org.openhab.binding.metofficedatahub.internal.dto.responses.SiteApiFeatureCollection;
import org.openhab.binding.metofficedatahub.internal.dto.responses.SiteApiFeatureProperties;
import org.openhab.binding.metofficedatahub.internal.dto.responses.SiteApiTimeSeries;
import org.openhab.core.i18n.LocaleProvider;
import org.openhab.core.i18n.LocationProvider;
import org.openhab.core.i18n.TimeZoneProvider;
import org.openhab.core.i18n.TranslationProvider;
import org.openhab.core.library.types.DateTimeType;
import org.openhab.core.library.types.DecimalType;
import org.openhab.core.library.types.PointType;
import org.openhab.core.library.types.QuantityType;
import org.openhab.core.library.unit.SIUnits;
import org.openhab.core.library.unit.Units;
import org.openhab.core.thing.Bridge;
import org.openhab.core.thing.ChannelUID;
import org.openhab.core.thing.Thing;
import org.openhab.core.thing.ThingStatus;
import org.openhab.core.thing.ThingStatusDetail;
import org.openhab.core.thing.binding.BaseThingHandler;
import org.openhab.core.types.Command;
import org.openhab.core.types.RefreshType;
import org.openhab.core.types.State;
import org.openhab.core.types.UnDefType;
import org.osgi.framework.Bundle;
import org.osgi.framework.FrameworkUtil;
import org.osgi.service.component.annotations.Reference;
/**
* The {@link MetOfficeDataHubSiteHandler} is responsible for handling commands, which are
* sent to one of the channels.
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public class MetOfficeDataHubSiteHandler extends BaseThingHandler implements ISiteResponseListener {
private final Object checkDataRequiredSchedulerLock = new Object();
private final Object checkDailySchedulerLock = new Object();
private final TranslationProvider translationProvider;
private final LocaleProvider localeProvider;
private final Bundle bundle;
private final LocationProvider locationProvider;
private final PollManager dailyForecastPollManager;
private final PollManager hourlyForecastPollManager;
private volatile MetOfficeDataHubSiteConfiguration config = getConfigAs(MetOfficeDataHubSiteConfiguration.class);
private PointType location = new PointType();
private @Nullable ScheduledFuture<?> checkDataRequiredScheduler = null;
private @Nullable ScheduledFuture<?> dailyScheduler = null;
private @Nullable ScheduledFuture<?> initTask = null;
private String dailyPollKey = "";
private String hourlyPollKey = "";
public MetOfficeDataHubSiteHandler(Thing thing, @Reference LocationProvider locationProvider,
@Reference TranslationProvider translationProvider, @Reference LocaleProvider localeProvider,
@Reference TimeZoneProvider timeZoneProvider) {
super(thing);
this.locationProvider = locationProvider;
this.translationProvider = translationProvider;
this.localeProvider = localeProvider;
this.bundle = FrameworkUtil.getBundle(getClass());
final ResponseDataProcessor updateHourlyFromCache = new ResponseDataProcessor() {
@Override
public void processResponse(final String content) {
processHourlyContent(content);
}
};
final ResponseDataProcessor updateDailyFromCache = new ResponseDataProcessor() {
@Override
public void processResponse(final String content) {
processDailyContent(content);
}
};
this.hourlyForecastPollManager = new PollManager("Hourly", timeZoneProvider, scheduler, Duration.ofHours(1),
updateHourlyFromCache, () -> {
sendForecastRequest(false);
});
this.dailyForecastPollManager = new PollManager("Daily", timeZoneProvider, scheduler, Duration.ofHours(3),
updateDailyFromCache, () -> {
sendForecastRequest(true);
});
}
@Override
public void dispose() {
cancelInitTask();
cancelDataRequiredCheck();
cancelScheduleDailyDataPoll(true);
hourlyForecastPollManager.dispose();
dailyForecastPollManager.dispose();
super.dispose();
}
@Override
public void initialize() {
updateStatus(ThingStatus.UNKNOWN);
dailyForecastPollManager.setDataRequired(false, false);
hourlyForecastPollManager.setDataRequired(false, false);
config = getConfigAs(MetOfficeDataHubSiteConfiguration.class);
if (config.location.isBlank()) {
@Nullable
PointType userLocation = locationProvider.getLocation();
if (userLocation == null) {
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
getLocalizedText("site.error.no-user-location"));
return;
} else {
location = userLocation;
}
} else {
try {
location = new PointType(config.location);
} catch (Exception e) {
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
getLocalizedText("site.error.invalid-location"));
return;
}
}
dailyPollKey = location + ",daily";
hourlyPollKey = location + ",hourly";
if (config.hourlyForecastPollRate > 0) {
hourlyForecastPollManager.setPollDuration(Duration.ofHours(config.hourlyForecastPollRate));
}
if (config.dailyForecastPollRate > 0) {
dailyForecastPollManager.setPollDuration(Duration.ofHours(config.dailyForecastPollRate));
}
scheduleInitTask();
}
@Override
public void channelLinked(ChannelUID channelUID) {
super.channelLinked(channelUID);
handleCommand(channelUID, RefreshType.REFRESH);
}
@Override
public void channelUnlinked(ChannelUID channelUID) {
// can be overridden by subclasses
scheduleDataRequiredCheck();
}
@Override
public void handleCommand(ChannelUID channelUID, Command command) {
scheduler.execute(() -> {
if (RefreshType.REFRESH.equals(command)) {
scheduleDataRequiredCheck();
}
});
}
private @Nullable MetOfficeDataHubBridgeHandler getMetOfficeDataHubBridge() {
Bridge baseBridge = getBridge();
if (baseBridge != null && baseBridge.getHandler() instanceof MetOfficeDataHubBridgeHandler bridgeHandler) {
return bridgeHandler;
} else {
return null;
}
}
protected boolean requiresPoll() {
return hourlyForecastPollManager.getIsDataRequired() || dailyForecastPollManager.getIsDataRequired();
}
protected void scheduleInitTask() {
cancelInitTask();
initTask = scheduler.schedule(() -> {
MetOfficeDataHubBridgeHandler metBridge = getMetOfficeDataHubBridge();
if (metBridge != null) {
updateStatus(metBridge.getThing().getStatus());
}
checkDataRequired();
}, 200, TimeUnit.MILLISECONDS);
}
private void cancelInitTask() {
final ScheduledFuture<?> initTaskRef = initTask;
if (initTaskRef != null) {
initTaskRef.cancel(true);
initTask = null;
}
}
private void checkDataRequired() {
final List<@Nullable String> activeGroups = getThing().getChannels().stream().filter(x -> isLinked(x.getUID()))
.map(x -> x.getUID().getGroupId()).distinct().toList();
if (activeGroups.stream().anyMatch(g -> g != null && g.startsWith(GROUP_PREFIX_DAILY_FORECAST))) {
dailyForecastPollManager.setDataRequired(true, true);
} else {
dailyForecastPollManager.setDataRequired(false, false);
}
if (activeGroups.stream().anyMatch(g -> g != null && g.startsWith(GROUP_PREFIX_HOURS_FORECAST))) {
hourlyForecastPollManager.setDataRequired(true, true);
} else {
hourlyForecastPollManager.setDataRequired(false, false);
}
}
private void sendForecastRequest(final boolean daily) {
MetOfficeDataHubBridgeHandler uplinkBridge = getMetOfficeDataHubBridge();
if (uplinkBridge == null) {
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
getLocalizedText("site.error.no-bridge"));
return;
}
final String pollId = (daily) ? dailyPollKey : hourlyPollKey;
final MetOfficeDataHubBridgeHandler metOfficeBridgeHandler = getMetOfficeDataHubBridge();
if (metOfficeBridgeHandler != null) {
if (!metOfficeBridgeHandler.getSiteApi().sendRequest(daily, location, this, pollId)) {
if (daily) {
dailyForecastPollManager.cachedPollOrLiveStart(false);
} else {
hourlyForecastPollManager.cachedPollOrLiveStart(false);
}
}
}
}
/**
* Scheduler to evaluate which data is required to be polled from
* the APIs
*/
private void scheduleDataRequiredCheck() {
synchronized (checkDataRequiredSchedulerLock) {
cancelDataRequiredCheck();
checkDataRequiredScheduler = scheduler.schedule(this::checkDataRequired, 2, TimeUnit.SECONDS);
}
}
private void cancelDataRequiredCheck() {
synchronized (checkDataRequiredSchedulerLock) {
ScheduledFuture<?> job = checkDataRequiredScheduler;
if (job != null) {
job.cancel(true);
checkDataRequiredScheduler = null;
}
}
}
private void cancelScheduleDailyDataPoll(final boolean allowInterrupt) {
synchronized (checkDailySchedulerLock) {
ScheduledFuture<?> job = dailyScheduler;
if (job != null) {
job.cancel(allowInterrupt);
dailyScheduler = null;
}
}
}
// Localization functionality
public String getLocalizedText(String key, @Nullable Object @Nullable... arguments) {
String result = translationProvider.getText(bundle, key, key, localeProvider.getLocale(), arguments);
return Objects.nonNull(result) ? result : key;
}
// Implementation of ISiteResponseListener and associated methods
@Override
public void processDailyResponse(final String responseData, final String pollId) {
if (dailyPollKey.equals(pollId)) {
dailyForecastPollManager.setDataContentReceived(responseData);
processDailyContent(responseData);
}
}
public void processDailyContent(final String responseData) {
final SiteApiFeatureCollection response = GSON.fromJson(responseData, SiteApiFeatureCollection.class);
if (response == null) {
return;
}
final SiteApiFeatureProperties props = response.getFirstProperties();
if (props == null) {
return;
}
final String startOfHour = MetOfficeDataHubSiteHandler.getDataTsAtLastChronoUnit(ChronoUnit.DAYS);
final int forecastForthisHour = props.getHourlyTimeSeriesPositionForCurrentHour(startOfHour);
for (int dayOffset = 0; dayOffset <= 6; ++dayOffset) {
// Calculate the correct array position for the data
final int dataIdx = (forecastForthisHour != -1) ? forecastForthisHour + dayOffset : -1;
final String channelPrefix = MetOfficeDataHubSiteHandler.calculatePrefix(GROUP_PREFIX_DAILY_FORECAST,
dayOffset);
final SiteApiTimeSeries data = props.getTimeSeries(dataIdx);
updateState(channelPrefix + SITE_TIMESTAMP, getDateTimeTypeState(data.getTime()));
updateState(channelPrefix + SITE_DAILY_MIDDAY_WIND_SPEED_10M,
getQuantityTypeState(data.getMidday10MWindSpeed(), Units.METRE_PER_SECOND));
updateState(channelPrefix + SITE_DAILY_MIDNIGHT_WIND_SPEED_10M,
getQuantityTypeState(data.getMidnight10MWindSpeed(), Units.METRE_PER_SECOND));
updateState(channelPrefix + SITE_DAILY_MIDDAY_WIND_DIRECTION_10M,
getQuantityTypeState(data.getMidday10MWindDirection(), Units.DEGREE_ANGLE));
updateState(channelPrefix + SITE_DAILY_MIDNIGHT_WIND_DIRECTION_10M,
getQuantityTypeState(data.getMidnight10MWindDirection(), Units.DEGREE_ANGLE));
updateState(channelPrefix + SITE_DAILY_MIDDAY_WIND_GUST_10M,
getQuantityTypeState(data.getMidday10MWindGust(), Units.METRE_PER_SECOND));
updateState(channelPrefix + SITE_DAILY_MIDNIGHT_WIND_GUST_10M,
getQuantityTypeState(data.getMidnight10MWindGust(), Units.METRE_PER_SECOND));
updateState(channelPrefix + SITE_DAILY_MIDDAY_VISIBILITY,
getQuantityTypeState(data.getMiddayVisibility(), METRE));
updateState(channelPrefix + SITE_DAILY_MIDNIGHT_VISIBILITY,
getQuantityTypeState(data.getMidnightVisibility(), METRE));
updateState(channelPrefix + SITE_DAILY_MIDDAY_REL_HUMIDITY,
getQuantityTypeState(data.getMiddayRelativeHumidity(), Units.PERCENT));
updateState(channelPrefix + SITE_DAILY_MIDNIGHT_REL_HUMIDITY,
getQuantityTypeState(data.getMidnightRelativeHumidity(), Units.PERCENT));
updateState(channelPrefix + SITE_DAILY_MIDDAY_PRESSURE,
getQuantityTypeState(data.getMiddayPressure(), SIUnits.PASCAL));
updateState(channelPrefix + SITE_DAILY_MIDNIGHT_PRESSURE,
getQuantityTypeState(data.getMidnightPressure(), SIUnits.PASCAL));
updateState(channelPrefix + SITE_DAILY_DAY_MAX_UV_INDEX, getDecimalTypeState(data.getMaxUvIndex()));
updateState(channelPrefix + SITE_DAILY_DAY_UPPER_BOUND_MAX_TEMP,
getQuantityTypeState(data.getDayUpperBoundMaxTemp(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_DAILY_DAY_LOWER_BOUND_MAX_TEMP,
getQuantityTypeState(data.getDayLowerBoundMaxTemp(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_DAILY_NIGHT_UPPER_BOUND_MAX_TEMP,
getQuantityTypeState(data.getNightUpperBoundMinTemp(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_DAILY_NIGHT_LOWER_BOUND_MAX_TEMP,
getQuantityTypeState(data.getNightLowerBoundMinTemp(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_DAILY_NIGHT_FEELS_LIKE_MIN_TEMP,
getQuantityTypeState(data.getNightMinFeelsLikeTemp(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_DAILY_DAY_FEELS_LIKE_MAX_TEMP,
getQuantityTypeState(data.getDayMaxFeelsLikeTemp(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_DAILY_NIGHT_LOWER_BOUND_MIN_TEMP,
getQuantityTypeState(data.getNightLowerBoundMinTemp(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_DAILY_DAY_MAX_FEELS_LIKE_TEMP,
getQuantityTypeState(data.getDayMaxFeelsLikeTemp(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_DAILY_NIGHT_LOWER_BOUND_MIN_FEELS_LIKE_TEMP,
getQuantityTypeState(data.getNightLowerBoundMinFeelsLikeTemp(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_DAILY_DAY_LOWER_BOUND_MAX_FEELS_LIKE_TEMP,
getQuantityTypeState(data.getDayLowerBoundMaxFeelsLikeTemp(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_DAILY_DAY_UPPER_BOUND_MAX_FEELS_LIKE_TEMP,
getQuantityTypeState(data.getDayUpperBoundMaxFeelsLikeTemp(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_DAILY_UPPER_BOUND_MIN_FEELS_LIKE_TEMP,
getQuantityTypeState(data.getNightUpperBoundMinFeelsLikeTemp(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_DAILY_DAY_MAX_SCREEN_TEMPERATURE,
getQuantityTypeState(data.getDayMaxScreenTemperature(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_DAILY_NIGHT_MIN_SCREEN_TEMPERATURE,
getQuantityTypeState(data.getNightMinScreenTemperature(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_DAILY_DAY_PROBABILITY_OF_PRECIPITATION,
getQuantityTypeState(data.getDayProbabilityOfPrecipitation(), Units.PERCENT));
updateState(channelPrefix + SITE_DAILY_NIGHT_PROBABILITY_OF_PRECIPITATION,
getQuantityTypeState(data.getNightProbabilityOfPrecipitation(), Units.PERCENT));
updateState(channelPrefix + SITE_DAILY_DAY_PROBABILITY_OF_SNOW,
getQuantityTypeState(data.getDayProbabilityOfSnow(), Units.PERCENT));
updateState(channelPrefix + SITE_DAILY_NIGHT_PROBABILITY_OF_SNOW,
getQuantityTypeState(data.getNightProbabilityOfSnow(), Units.PERCENT));
updateState(channelPrefix + SITE_DAILY_DAY_PROBABILITY_OF_HEAVY_SNOW,
getQuantityTypeState(data.getDayProbabilityOfHeavySnow(), Units.PERCENT));
updateState(channelPrefix + SITE_DAILY_NIGHT_PROBABILITY_OF_HEAVY_SNOW,
getQuantityTypeState(data.getNightProbabilityOfHeavySnow(), Units.PERCENT));
updateState(channelPrefix + SITE_DAILY_DAY_PROBABILITY_OF_RAIN,
getQuantityTypeState(data.getDayProbabilityOfRain(), Units.PERCENT));
updateState(channelPrefix + SITE_DAILY_NIGHT_PROBABILITY_OF_RAIN,
getQuantityTypeState(data.getNightProbabilityOfRain(), Units.PERCENT));
updateState(channelPrefix + SITE_DAILY_DAY_PROBABILITY_OF_HEAVY_RAIN,
getQuantityTypeState(data.getDayProbabilityOfHeavyRain(), Units.PERCENT));
updateState(channelPrefix + SITE_DAILY_NIGHT_PROBABILITY_OF_HEAVY_RAIN,
getQuantityTypeState(data.getNightProbabilityOfHeavyRain(), Units.PERCENT));
updateState(channelPrefix + SITE_DAILY_DAY_PROBABILITY_OF_HAIL,
getQuantityTypeState(data.getDayProbabilityOfHail(), Units.PERCENT));
updateState(channelPrefix + SITE_DAILY_NIGHT_PROBABILITY_OF_HAIL,
getQuantityTypeState(data.getNightProbabilityOfHail(), Units.PERCENT));
updateState(channelPrefix + SITE_DAILY_DAY_PROBABILITY_OF_SFERICS,
getQuantityTypeState(data.getDayProbabilityOfSferics(), Units.PERCENT));
updateState(channelPrefix + SITE_DAILY_NIGHT_PROBABILITY_OF_SFERICS,
getQuantityTypeState(data.getNightProbabilityOfSferics(), Units.PERCENT));
}
}
@Override
public void processHourlyResponse(final String responseData, final String pollId) {
if (hourlyPollKey.equals(pollId)) {
hourlyForecastPollManager.setDataContentReceived(responseData);
processHourlyContent(responseData);
}
}
public void processHourlyContent(final String responseData) {
final SiteApiFeatureCollection response = GSON.fromJson(responseData, SiteApiFeatureCollection.class);
if (response == null) {
return;
}
final SiteApiFeatureProperties props = response.getFirstProperties();
if (props == null) {
return;
}
final String startOfHour = MetOfficeDataHubSiteHandler.getDataTsAtLastChronoUnit(ChronoUnit.HOURS);
final int forecastForthisHour = props.getHourlyTimeSeriesPositionForCurrentHour(startOfHour);
for (int hrOffset = 0; hrOffset <= 24; ++hrOffset) {
// Calculate the correct array position for the data
final int dataIdx = (forecastForthisHour != -1) ? forecastForthisHour + hrOffset : -1;
final SiteApiTimeSeries data = props.getTimeSeries(dataIdx);
final String channelPrefix = MetOfficeDataHubSiteHandler.calculatePrefix(GROUP_PREFIX_HOURS_FORECAST,
hrOffset);
updateState(channelPrefix + SITE_TIMESTAMP, getDateTimeTypeState(data.getTime()));
updateState(channelPrefix + SITE_HOURLY_FORECAST_SCREEN_TEMPERATURE,
getQuantityTypeState(data.getScreenTemperature(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_HOURLY_FORECAST_MIN_SCREEN_TEMPERATURE,
getQuantityTypeState(data.getMinScreenTemperature(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_HOURLY_FORECAST_MAX_SCREEN_TEMPERATURE,
getQuantityTypeState(data.getMaxScreenTemperature(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_HOURLY_FEELS_LIKE_TEMPERATURE,
getQuantityTypeState(data.getFeelsLikeTemperature(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_HOURLY_SCREEN_RELATIVE_HUMIDITY,
getQuantityTypeState(data.getScreenRelativeHumidity(), Units.PERCENT));
updateState(channelPrefix + SITE_HOURLY_VISIBILITY, getQuantityTypeState(data.getVisibility(), METRE));
updateState(channelPrefix + SITE_HOURLY_PROBABILITY_OF_PRECIPITATION,
getQuantityTypeState(data.getProbOfPrecipitation(), Units.PERCENT));
updateState(channelPrefix + SITE_HOURLY_PRECIPITATION_RATE,
getQuantityTypeState(data.getPrecipitationRate(), Units.MILLIMETRE_PER_HOUR));
updateState(channelPrefix + SITE_HOURLY_TOTAL_PRECIPITATION_AMOUNT,
getQuantityTypeState(data.getTotalPrecipAmount(), MILLI(METRE)));
updateState(channelPrefix + SITE_HOURLY_TOTAL_SNOW_AMOUNT,
getQuantityTypeState(data.getTotalSnowAmount(), MILLI(METRE)));
updateState(channelPrefix + SITE_HOURLY_PRESSURE, getQuantityTypeState(data.getPressure(), SIUnits.PASCAL));
updateState(channelPrefix + SITE_HOURLY_WIND_SPEED_10M,
getQuantityTypeState(data.getWindSpeed10m(), Units.METRE_PER_SECOND));
updateState(channelPrefix + SITE_HOURLY_MAX_10M_WIND_GUST,
getQuantityTypeState(data.getMax10mWindGust(), Units.METRE_PER_SECOND));
updateState(channelPrefix + SITE_HOURLY_WIND_GUST_SPEED_10M,
getQuantityTypeState(data.getWindGustSpeed10m(), Units.METRE_PER_SECOND));
updateState(channelPrefix + SITE_HOURLY_SCREEN_DEW_POINT_TEMPERATURE,
getQuantityTypeState(data.getScreenDewPointTemperature(), SIUnits.CELSIUS));
updateState(channelPrefix + SITE_HOURLY_UV_INDEX, getDecimalTypeState(data.getUvIndex()));
updateState(channelPrefix + SITE_HOURLY_WIND_DIRECTION_FROM_10M,
getQuantityTypeState(data.getWindDirectionFrom10m(), Units.DEGREE_ANGLE));
}
}
public static String getDataTsAtLastChronoUnit(final ChronoUnit unit) {
return Instant.now().truncatedTo(unit).toString().substring(0, 16) + "Z";
}
// Helpers for updating channels support
private static String calculatePrefix(final String prefix, final int plusOffset) {
final StringBuilder strBldr = new StringBuilder(26);
strBldr.append(prefix);
if (plusOffset > 0) {
strBldr.append(GROUP_POSTFIX_BOTH_FORECASTS);
if (plusOffset < 10) {
strBldr.append("0");
}
strBldr.append(plusOffset);
}
strBldr.append(GROUP_PREFIX_TO_ITEM);
return strBldr.toString();
}
protected State getDateTimeTypeState(@Nullable String value) {
return (value == null) ? UnDefType.UNDEF : new DateTimeType(value).toLocaleZone();
}
protected State getQuantityTypeState(@Nullable Number value, Unit<?> unit) {
return (value == null) ? UnDefType.UNDEF : new QuantityType<>(value, unit);
}
protected State getDecimalTypeState(@Nullable Number value) {
return (value == null) ? UnDefType.UNDEF : new DecimalType(value);
}
}

View File

@ -0,0 +1,55 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
/**
* The {@link MetOfficeDelayedExecutor} wraps up the executor functionality for a delayed execution with the
* relevant locking.
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public class MetOfficeDelayedExecutor {
public MetOfficeDelayedExecutor(final ScheduledExecutorService scheduler) {
this.scheduler = scheduler;
}
private final ScheduledExecutorService scheduler;
private final Object scheduledFutureRefLock = new Object();
private @Nullable ScheduledFuture<?> scheduledFutureRef = null;
public void scheduleExecution(final long initialDelay, final Runnable task) {
synchronized (scheduledFutureRefLock) {
cancelScheduledTask(true);
scheduledFutureRef = scheduler.schedule(task, initialDelay, TimeUnit.MILLISECONDS);
}
}
public void cancelScheduledTask(final boolean allowInterrupt) {
synchronized (scheduledFutureRefLock) {
ScheduledFuture<?> job = scheduledFutureRef;
if (job != null) {
job.cancel(true);
scheduledFutureRef = null;
}
}
}
}

View File

@ -0,0 +1,204 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal;
import static org.openhab.binding.metofficedatahub.internal.MetOfficeDataHubBindingConstants.DAY_IN_MILLIS;
import static org.openhab.binding.metofficedatahub.internal.MetOfficeDataHubBindingConstants.EXPECTED_TS_FORMAT;
import static org.openhab.binding.metofficedatahub.internal.MetOfficeDataHubBindingConstants.RANDOM_GENERATOR;
import java.time.Duration;
import java.time.Instant;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.ZonedDateTime;
import java.time.format.DateTimeFormatter;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledFuture;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
import org.openhab.binding.metofficedatahub.internal.api.ResponseDataProcessor;
import org.openhab.core.i18n.TimeZoneProvider;
import org.osgi.service.component.annotations.Reference;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* The {@link PollManager} manages basic poll management functionality.
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public class PollManager {
private volatile long lastForecastPoll = -1;
private final Logger logger = LoggerFactory.getLogger(PollManager.class);
private final String pollName;
ResponseDataProcessor runCachedDataPoll;
Runnable runLiveDataPoll;
Duration durationBetweenPolls;
String lastRepsonse = "";
private @Nullable ScheduledFuture<?> pollScheduled = null;
private final Object pollScheduledLock = new Object();
private final TimeZoneProvider timeZoneProvider;
/**
* This handles the scheduling of an hourly forecast poll, to be applied with the given delay.
* When run, if requests the run-time of the next one is calculated and scheduled.
*/
private final MetOfficeDelayedExecutor forecastJob;
private volatile boolean dataRequired;
public PollManager(final String pollName, @Reference TimeZoneProvider timeZoneProvider,
@Reference ScheduledExecutorService scheduler, final Duration durationBetweenPolls,
final ResponseDataProcessor cachedDataPoll, final Runnable liveDataPoll) {
this.pollName = pollName;
this.durationBetweenPolls = durationBetweenPolls;
this.runCachedDataPoll = cachedDataPoll;
this.runLiveDataPoll = liveDataPoll;
this.timeZoneProvider = timeZoneProvider;
forecastJob = new MetOfficeDelayedExecutor(scheduler);
dataRequired = false;
}
public void dispose() {
cancelScheduledPoll(true);
forecastJob.cancelScheduledTask(true);
}
public void cachedPollOrLiveStart(final boolean attemptLivePollIfRequired) {
if (dataRequired) {
if (!lastRepsonse.isEmpty()) {
logger.trace("Using cached {} forecast response data", pollName);
runCachedDataPoll.processResponse(lastRepsonse);
} else {
logger.trace("Starting poll sequence for {} forecast data", pollName);
if (attemptLivePollIfRequired) {
reconfigurePolling();
}
}
} else {
logger.trace("Skipping refresh on non-required data for {} forecast", pollName);
}
}
public void setPollDuration(final Duration durationBetweenPolls) {
this.durationBetweenPolls = durationBetweenPolls;
}
public void setDataRequired(final boolean dataRequired, final boolean reconfigureIfRequired) {
final boolean previousValue = this.dataRequired;
this.dataRequired = dataRequired;
if (dataRequired) {
logger.trace("{} data poll required", pollName);
} else {
logger.trace("{} data poll not required", pollName);
}
if (dataRequired) {
final boolean reconfigureRequired = reconfigureIfRequired && !previousValue;
if (reconfigureRequired) {
reconfigurePolling();
}
cachedPollOrLiveStart(!reconfigureRequired);
}
}
public boolean getIsDataRequired() {
return dataRequired;
}
public void reconfigurePolling() {
final long millisSinceDayStart = getMillisSinceDayStart();
final long pollRateMillis = durationBetweenPolls.toMillis();
final long initialDelayTimeToFirstCycle = pollRateMillis - (millisSinceDayStart % pollRateMillis);
final long lastPollExpectedTime = System.currentTimeMillis() - (pollRateMillis - initialDelayTimeToFirstCycle);
logger.trace("Last {} poll expected time should have been : {}", pollName,
millisToLocalDateTime(lastPollExpectedTime));
logger.trace("Last {} poll time should have been : {}", pollName, millisToLocalDateTime(lastForecastPoll));
// Poll if a poll hasn't been done before, or if the previous poll was before what would be now the new
// poll intervals last poll time then a poll should be run now.
if (lastForecastPoll == -1 || lastPollExpectedTime > lastForecastPoll) {
liveDataPoll();
} else {
if (dataRequired && !lastRepsonse.isEmpty()) {
runCachedDataPoll.processResponse(lastRepsonse);
}
}
scheduleNextPoll();
}
private void cancelScheduledPoll(final boolean allowInterrupt) {
synchronized (pollScheduledLock) {
ScheduledFuture<?> job = pollScheduled;
if (job != null) {
job.cancel(allowInterrupt);
pollScheduled = null;
}
}
}
public void setDataContentReceived(final String responseContent) {
this.lastRepsonse = responseContent;
}
protected static long getMillisSinceDayStart() {
return Duration.between(LocalDate.now().atStartOfDay(), LocalDateTime.now()).toMillis();
}
private String millisToLocalDateTime(final long milliseconds) {
ZonedDateTime cvDate = Instant.ofEpochMilli(milliseconds).atZone(timeZoneProvider.getTimeZone());
return cvDate.format(DateTimeFormatter.ofPattern(EXPECTED_TS_FORMAT));
}
private void scheduleNextPoll() {
final long millisSinceDayStart = getMillisSinceDayStart();
long pollRateMillis = durationBetweenPolls.toMillis();
long initialDelayTimeToFirstCycle = pollRateMillis - (millisSinceDayStart % pollRateMillis);
if (initialDelayTimeToFirstCycle + millisSinceDayStart > DAY_IN_MILLIS) {
logger.debug("Not scheduling {} poll after next daily cycle reset", pollName);
} else {
logger.debug("Scheduling next {} forecast data poll to be in {} milliseconds at {}", pollName,
initialDelayTimeToFirstCycle,
millisToLocalDateTime(System.currentTimeMillis() + initialDelayTimeToFirstCycle));
initialDelayTimeToFirstCycle += RANDOM_GENERATOR.nextInt(60000);
// Schedule the first poll to occur after the given delay
forecastJob.scheduleExecution(initialDelayTimeToFirstCycle, () -> {
liveDataPoll();
scheduleNextPoll();
});
}
}
public void liveDataPoll() {
if (getIsDataRequired()) {
logger.debug("Doing a POLL for the {} forecast", pollName);
runLiveDataPoll.run();
} else {
logger.debug("Skipping a POLL for the {} forecast", pollName);
}
};
}

View File

@ -0,0 +1,49 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.api;
import java.io.Serial;
import org.eclipse.jdt.annotation.NonNullByDefault;
/**
* The {@link AuthTokenException} should be thrown when the endpoint being communicated with
* does not appear to be a Tap Link Gateway device.
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public class AuthTokenException extends I18Exception {
@Serial
private static final long serialVersionUID = -7786449325604153947L;
public AuthTokenException() {
super();
}
public AuthTokenException(final String message) {
super(message);
}
public AuthTokenException(final Throwable cause) {
super(cause);
}
public AuthTokenException(final String message, final Throwable cause) {
super(message, cause);
}
public String getI18Key() {
return getI18Key("exception.bad-auth-token");
}
}

View File

@ -0,0 +1,58 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.api;
import java.io.Serial;
import java.util.Objects;
import org.eclipse.jdt.annotation.NonNullByDefault;
/**
* The {@link I18Exception} is a abstract class for exceptions that support
* i18key functionality.
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public abstract class I18Exception extends Exception {
@Serial
private static final long serialVersionUID = -7784829349743963947L;
protected String i18Key = "";
public I18Exception() {
super();
}
public I18Exception(final String message) {
super(message);
}
public I18Exception(final Throwable cause) {
super(cause);
}
public I18Exception(final String message, final Throwable cause) {
super(message, cause);
}
public abstract String getI18Key();
public String getI18Key(final String defaultI18) {
if (!i18Key.isBlank()) {
return i18Key;
}
return Objects.requireNonNullElse(getMessage(), defaultI18);
}
}

View File

@ -0,0 +1,47 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.api;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
/**
* Implementations of this interface, allow access to a HttpClient which can be used
* for communication requests to LinkTap Gateways.
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public interface IConnectionStatusListener {
/**
* This is invoked to notify implementations of this interface, whether the connection has been
* successfully authenticated or given a response indicating an authentication failure, or that
* the authentication data is not in a valid format.
*
* @param authenticated is true when the authentication was validated and successful used.
*/
void processAuthenticationResult(final boolean authenticated);
/**
* This is invoked to notify implementations of this interface, upon a failure of communications.
*
* @param t is instance of the Throwable that was raised/thrown when the communications failed to process.
*/
void processCommunicationFailure(final @Nullable Throwable t);
/**
* This is invoked to notify implementations of this interface, when connectivity has been successful.
*/
void processConnected();
}

View File

@ -0,0 +1,33 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.api;
import org.eclipse.jdt.annotation.NonNullByDefault;
/**
* Implementations of this interface, allow the monitoring of when the rate limiter
* has updated its operating parameters.
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public interface IRateLimiterListener {
/**
* This is invoked to notify implementations of this interface, that the given rate limiter
* has been updated, with new counts.
*
* @param requestLimiter is a reference to the rate limiter that has been updated.
*/
void processRateLimiterUpdated(final RequestLimiter requestLimiter);
}

View File

@ -0,0 +1,45 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.api;
import org.eclipse.jdt.annotation.NonNullByDefault;
/**
* Implementations of this interface, allow the responses of a SiteAPI request to
* be processed
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public interface ISiteResponseListener {
/**
* This is invoked to notify implementations of this interface, new daily response data has been received.
* It is at the implementations discretion whether the data is of interest based on the pollId, which is set
* from when the original poll was requested.
*
* @param content is the daily response JSON content returned for a site API request.
* @param pollId is the ID associated to the request this was requested with.
*/
void processDailyResponse(final String content, final String pollId);
/**
* This is invoked to notify implementations of this interface, new hourly response data has been received.
* It is at the implementations discretion whether the data is of interest based on the pollId, which is set
* from when the original poll was requested.
*
* @param content is the hourly response JSON content returned for a site API request.
* @param pollId is the ID associated to the request this was requested with.
*/
void processHourlyResponse(final String content, final String pollId);
}

View File

@ -0,0 +1,46 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.api;
import org.eclipse.jdt.annotation.NonNullByDefault;
import com.google.gson.annotations.SerializedName;
/**
* The {@link JwtTokenHeader} allows the basic decoding of a JWT token header, to allow
* basic validation before using it.
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public class JwtTokenHeader {
@SerializedName("x5t")
private String x5t = "";
@SerializedName("kid")
private String kid = "";
@SerializedName("typ")
private String typ = "";
@SerializedName("alg")
private String alg = "";
public boolean isValid() {
if (x5t.isBlank() || kid.isBlank() || typ.isBlank() || alg.isBlank()) {
return false;
}
return "JWT".contentEquals(typ);
}
}

View File

@ -0,0 +1,46 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.api;
import org.eclipse.jdt.annotation.NonNullByDefault;
import com.google.gson.annotations.SerializedName;
/**
* The {@link JwtTokenPayload} allows the basic decoding of a JWT token header, to allow
* basic validation before using it.
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public class JwtTokenPayload {
@SerializedName("sub")
private String sub = "";
@SerializedName("iss")
private String iss = "";
@SerializedName("keytype")
private String keyType = "";
@SerializedName("token_type")
private String tokenType = "";
public boolean isValid() {
if (sub.isBlank() || iss.isBlank() || keyType.isBlank() || tokenType.isBlank()) {
return false;
}
return "apiKey".contentEquals(tokenType);
}
}

View File

@ -0,0 +1,97 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.api;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import javax.validation.constraints.NotNull;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
/**
* The {@link LookupWrapper} is a container providing common functionality for providing
* key -> T mappings. The backend store is ConcurrentHashMap.
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public class LookupWrapper<@Nullable itemT> {
final Map<@NotNull String, @Nullable itemT> storeLookup = new ConcurrentHashMap<>();
/**
* Register using key the given T instance, and after addition call the specified Runnable
*
* @param key - The key for the item
* @param item - The instance to store a reference to
* @param afterAddition - The runnable to run after the addition has been completed
* @return - false if another item is already assigned to the key preventing the addition, or true
* when added successfully.
*/
public boolean registerItem(final @NotNull String key, final @NotNull itemT item,
@NotNull final Runnable afterAddition) {
if (storeLookup.containsKey(key)) {
final itemT found = storeLookup.get(key);
if (found != null && !found.equals(item)) {
return false;
}
}
storeLookup.put(key, item);
afterAddition.run();
return true;
}
/**
* Remove the given key and item combination
*
* @param key - The expected key of the item
* @param item - The item referenced by the key
* @param whenEmpty - Runnable executed when no more key -> item mappings exist
*/
public void deregisterItem(final @NotNull String key, final @NotNull itemT item,
@NotNull final Runnable whenEmpty) {
storeLookup.remove(key, item);
if (storeLookup.isEmpty()) {
whenEmpty.run();
}
}
/**
* Returns the item associated to the given key
*
* @param key - the key to find the item for
* @return - null if no item is found otherwise the found item
*/
public @Nullable itemT getItem(final @NotNull String key) {
return storeLookup.get(key);
}
/**
* Clears a entry when only the given key is known
*
* @param key - the key remove if it exists
*/
public void clearItem(final @NotNull String key) {
storeLookup.remove(key);
}
/**
* Get an array of all entries stored at the time of calling
*/
public List<@NotNull itemT> getItemlist() {
return storeLookup.values().stream().toList();
}
}

View File

@ -0,0 +1,200 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.api;
import java.time.Instant;
import java.time.ZoneId;
import java.time.ZonedDateTime;
import java.time.format.DateTimeFormatter;
import java.time.format.DateTimeParseException;
import java.time.temporal.ChronoUnit;
import java.util.Objects;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
import org.openhab.core.i18n.LocaleProvider;
import org.openhab.core.i18n.TimeZoneProvider;
import org.openhab.core.i18n.TranslationProvider;
import org.openhab.core.storage.Storage;
import org.openhab.core.storage.StorageService;
import org.osgi.framework.Bundle;
import org.osgi.service.component.annotations.Reference;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* The {@link RequestLimiter} is responsible for handling commands, which are
* sent to one of the channels.
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public class RequestLimiter {
public static final int INVALID_REQUEST_ID = -1;
public static final int SECONDS_PER_DAY = 86400;
static final ZoneId UTC_ZONE_ID = ZoneId.of("UTC");
final StorageService storageService;
final ScheduledExecutorService scheduler;
final TimeZoneProvider timeZoneProvider;
final TranslationProvider translationProvider;
final LocaleProvider localeProvider;
final Bundle bundle;
private final Logger logger = LoggerFactory.getLogger(RequestLimiter.class);
private final Object dailyResetLock = new Object();
private int requestLimit = 0;
private int currentRequestCount = 0;
private String limiterId;
private String storageKeyCount;
private String storageKeyTimestamp;
private @Nullable ScheduledFuture<?> dailyResetFuture = null;
public int getCurrentRequestCount() {
return currentRequestCount;
}
public RequestLimiter(final String limiterId, @Reference StorageService storageService,
@Reference TimeZoneProvider timeZoneProvider, ScheduledExecutorService scheduler,
@Reference TranslationProvider translationProvider, @Reference LocaleProvider localeProvider,
@Reference Bundle bundle) {
this.limiterId = limiterId;
this.storageKeyCount = limiterId + "_count";
this.storageKeyTimestamp = limiterId + "_ts";
this.storageService = storageService;
this.timeZoneProvider = timeZoneProvider;
this.scheduler = scheduler;
this.translationProvider = translationProvider;
this.localeProvider = localeProvider;
this.bundle = bundle;
loadLimiterData();
}
public void dispose() {
cancelPendingDailyReset();
saveLimiterData();
}
private void scheduleStandardResetTime(final int resetLimit) {
// Met Office Data Hub resets at midnight UTC tz it's daily counter call's
final ZonedDateTime currentTs = Instant.now().atZone(UTC_ZONE_ID);
final ZonedDateTime resetTs = currentTs.minusHours(currentTs.getHour()).minusMinutes(currentTs.getMinute())
.minusSeconds(currentTs.getSecond()).minusNanos(currentTs.getNano()).plusDays(1);
scheduleDailyReset(resetTs, resetLimit);
}
private void cancelPendingDailyReset() {
synchronized (dailyResetLock) {
final ScheduledFuture<?> ref = dailyResetFuture;
if (ref != null) {
ref.cancel(true);
dailyResetFuture = null;
}
}
}
private void scheduleDailyReset(final ZonedDateTime resetLimiterDailyTime, final int resetLimit) {
final ZonedDateTime currentTs = Instant.now().atZone(resetLimiterDailyTime.getZone());
final long secondsUnitReset = ChronoUnit.SECONDS.between(currentTs, resetLimiterDailyTime);
synchronized (dailyResetLock) {
cancelPendingDailyReset();
dailyResetFuture = scheduler.scheduleWithFixedDelay(() -> {
resetLimiter(resetLimit);
}, secondsUnitReset, SECONDS_PER_DAY, TimeUnit.SECONDS);
}
}
private void loadLimiterData() {
final Storage<String> storage = storageService.getStorage(limiterId, String.class.getClassLoader());
@Nullable
final String countStored = storage.get(storageKeyCount);
@Nullable
final String tsStored = storage.get(storageKeyTimestamp);
if (countStored != null && tsStored != null) {
int newCount = -1;
try {
newCount = Integer.parseInt(countStored);
ZonedDateTime newTs = ZonedDateTime.parse(tsStored, DateTimeFormatter.ISO_ZONED_DATE_TIME);
final ZonedDateTime currentTs = Instant.now().atZone(UTC_ZONE_ID);
if (newTs.getDayOfYear() == currentTs.getDayOfYear()) {
currentRequestCount = newCount;
logger.trace("Limiter {} -> Restored Request Limiter count of {} for day of year {}", limiterId,
currentRequestCount, currentTs.getDayOfYear());
} else {
logger.trace("Limiter {} -> Days of saved data are not the same not restoring {} != {}", limiterId,
newTs.getDayOfYear(), currentTs.getDayOfYear());
}
} catch (DateTimeParseException | NumberFormatException exception) {
logger.warn("{}",
getLocalizedText("api.log.rate-limiter.failed-restore", limiterId, exception.getMessage()),
exception);
}
}
}
private void saveLimiterData() {
final ZonedDateTime saveTime = Instant.now().atZone(UTC_ZONE_ID);
final Storage<String> storage = storageService.getStorage(limiterId, String.class.getClassLoader());
storage.put(storageKeyCount, String.valueOf(currentRequestCount));
storage.put(storageKeyTimestamp, saveTime.format(DateTimeFormatter.ISO_INSTANT));
logger.trace("Limiter {} -> Persisted Request Limiter count of {} for date {}", limiterId, currentRequestCount,
saveTime.format(DateTimeFormatter.ISO_ZONED_DATE_TIME));
}
public void resetLimiter() {
resetLimiter(requestLimit);
}
public synchronized void resetLimiter(int newLimit) {
requestLimit = newLimit;
currentRequestCount = 0;
logger.trace("Limiter {} -> Resetting limiter to 0 used for new limit {}", limiterId, newLimit);
scheduler.schedule(this::saveLimiterData, 1, TimeUnit.SECONDS);
}
public synchronized void updateLimit(int newLimit) {
requestLimit = newLimit;
logger.trace("Limiter {} -> Updated limiter to new total limit {}", limiterId, newLimit);
scheduler.schedule(this::saveLimiterData, 1, TimeUnit.SECONDS);
scheduleStandardResetTime(newLimit);
}
public synchronized int getRequestCountIfAvailable() {
final int requestId = currentRequestCount;
if (currentRequestCount < requestLimit) {
++currentRequestCount;
scheduler.schedule(this::saveLimiterData, 1, TimeUnit.SECONDS);
} else {
return INVALID_REQUEST_ID;
}
return requestId;
}
public boolean isInvalidRequestId(final int requestId) {
return INVALID_REQUEST_ID == requestId;
}
// Localization functionality
public String getLocalizedText(String key, @Nullable Object @Nullable... arguments) {
String result = translationProvider.getText(bundle, key, key, localeProvider.getLocale(), arguments);
return Objects.nonNull(result) ? result : key;
}
}

View File

@ -0,0 +1,26 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.api;
import org.eclipse.jdt.annotation.NonNullByDefault;
/**
* Implementations of this interface, allow the responses of a SiteAPI request to
* be processed
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public abstract class ResponseDataProcessor {
public abstract void processResponse(final String content);
}

View File

@ -0,0 +1,267 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.api;
import static org.openhab.binding.metofficedatahub.internal.MetOfficeDataHubBindingConstants.*;
import java.util.Objects;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
import javax.ws.rs.core.HttpHeaders;
import javax.ws.rs.core.MediaType;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
import org.eclipse.jetty.client.HttpClient;
import org.eclipse.jetty.client.api.Request;
import org.eclipse.jetty.client.api.Response;
import org.eclipse.jetty.client.api.Result;
import org.eclipse.jetty.client.util.BufferingResponseListener;
import org.eclipse.jetty.http.HttpMethod;
import org.openhab.core.i18n.LocaleProvider;
import org.openhab.core.i18n.TimeZoneProvider;
import org.openhab.core.i18n.TranslationProvider;
import org.openhab.core.io.net.http.HttpClientFactory;
import org.openhab.core.library.types.PointType;
import org.openhab.core.storage.StorageService;
import org.osgi.framework.Bundle;
import org.osgi.framework.FrameworkUtil;
import org.osgi.service.component.annotations.Activate;
import org.osgi.service.component.annotations.Reference;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* This provides the communications layer for the Site API
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public class SiteApi {
protected final LookupWrapper<@Nullable IConnectionStatusListener> authenticationListeners = new LookupWrapper<>();
protected final LookupWrapper<@Nullable IRateLimiterListener> rateLimiterListeners = new LookupWrapper<>();
private final Logger logger = LoggerFactory.getLogger(SiteApi.class);
// Utilised for sending communications
private final HttpClient httpClient;
// Utilised for persistence of rate limiter data between reboots
private final ScheduledExecutorService scheduler;
private final SiteApiAuthentication apiAuth;
private final RequestLimiter requestLimiter;
private final String usageId;
private final TranslationProvider translationProvider;
private final LocaleProvider localeProvider;
private final Bundle bundle;
@Activate
public SiteApi(String usageId, @Reference HttpClientFactory httpClientFactory,
@Reference StorageService storageService, @Reference TranslationProvider translationProvider,
@Reference LocaleProvider localeProvider, @Reference TimeZoneProvider timeZoneProvider,
@Reference ScheduledExecutorService scheduler) {
this.usageId = usageId;
this.httpClient = httpClientFactory.getCommonHttpClient();
this.apiAuth = new SiteApiAuthentication();
this.translationProvider = translationProvider;
this.localeProvider = localeProvider;
this.bundle = FrameworkUtil.getBundle(getClass());
this.requestLimiter = new RequestLimiter(usageId, storageService, timeZoneProvider, scheduler,
translationProvider, localeProvider, bundle);
this.scheduler = scheduler;
}
public void registerListeners(final String id, final Object candidateListener) {
if (candidateListener instanceof IConnectionStatusListener connectionStatusListener) {
authenticationListeners.registerItem(id, connectionStatusListener, NO_OP);
}
if (candidateListener instanceof IRateLimiterListener rateLimiterListener) {
rateLimiterListeners.registerItem(id, rateLimiterListener, NO_OP);
}
}
public void deregisterListeners(final String id, final Object candidateListener) {
if (candidateListener instanceof IConnectionStatusListener connectionStatusListener) {
authenticationListeners.deregisterItem(id, connectionStatusListener, NO_OP);
}
if (candidateListener instanceof IRateLimiterListener rateLimiterListener) {
rateLimiterListeners.deregisterItem(id, rateLimiterListener, NO_OP);
}
}
public void dispose() {
requestLimiter.dispose();
apiAuth.dispose();
}
public void setLimits(final int maxDailyCallLimit) {
requestLimiter.updateLimit(maxDailyCallLimit);
}
public void setApiKey(final String apiKey) {
try {
apiAuth.setApiKey(apiKey);
} catch (AuthTokenException ate) {
notifyAuthenticationListeners(false);
}
}
private void notifyRateLimiterListeners() {
scheduler.execute(() -> {
rateLimiterListeners.getItemlist().forEach(rateLimiterListener -> {
if (rateLimiterListener != null) {
rateLimiterListener.processRateLimiterUpdated(requestLimiter);
}
});
});
}
private void notifyAuthenticationListeners(final boolean authenticated) {
scheduler.execute(() -> {
authenticationListeners.getItemlist().forEach(authListener -> {
if (authListener != null) {
authListener.processAuthenticationResult(authenticated);
}
});
});
}
private void notifyCommFailureListeners(final @Nullable Throwable e) {
scheduler.execute(() -> {
authenticationListeners.getItemlist().forEach(authListener -> {
if (authListener != null) {
authListener.processCommunicationFailure(e);
}
});
});
}
private void notifyConnectedListeners() {
scheduler.execute(() -> {
authenticationListeners.getItemlist().forEach(authListener -> {
if (authListener != null) {
authListener.processConnected();
}
});
});
}
public void validateSiteApi() {
final PointType siteApiTestLocation = new PointType("51.5072,0.1276");
final Response.CompleteListener siteResponseListener = new BufferingResponseListener() { // 4.5kb buffer
@Override
public void onComplete(@Nullable Result result) {
if (result != null && !result.isFailed()) {
notifyConnectedListeners();
} else {
if (result != null) {
notifyCommFailureListeners(result.getFailure());
} else {
notifyCommFailureListeners(new Throwable("Unknown"));
}
}
}
};
if (requestLimiter.getRequestCountIfAvailable() == RequestLimiter.INVALID_REQUEST_ID) {
logger.warn("{}", getLocalizedText("comm.comm-check.no-quota-left"));
notifyConnectedListeners();
return;
}
sendAsyncSiteApiRequest(true, siteApiTestLocation, siteResponseListener, "validateSiteApiCheck");
}
public boolean sendRequest(final boolean daily, final PointType location,
final ISiteResponseListener siteResponseListener, final String pollId) {
if (requestLimiter.getRequestCountIfAvailable() == RequestLimiter.INVALID_REQUEST_ID) {
logger.debug("{} - Disabled requesting data - request limit has been hit", usageId);
return false;
}
notifyRateLimiterListeners();
final Response.CompleteListener listener = new BufferingResponseListener() { // 4.5kb buffer will cover both
@Override
public void onComplete(@Nullable Result result) {
if (result != null) {
final boolean userAuthValidatedPreviously = apiAuth.getIsAuthenticated();
apiAuth.processResult(result);
if (!apiAuth.getIsAuthenticated()) {
// Callback Async function confirming authorization is completed.
notifyAuthenticationListeners(false);
return;
}
// User is authorized at this point
if (!userAuthValidatedPreviously) {
// Authorization token confirmed
// Callback Async function confirming authorization is completed.
notifyAuthenticationListeners(true);
}
if (result.isSucceeded()) {
final String response = getContentAsString();
if (response != null) {
logger.trace("Got response for poll ID: \"{}\"", pollId);
notifyConnectedListeners();
scheduler.execute(() -> {
if (daily) {
siteResponseListener.processDailyResponse(response, pollId);
} else {
siteResponseListener.processHourlyResponse(response, pollId);
}
});
}
} else {
notifyCommFailureListeners(result.getFailure());
}
}
}
};
sendAsyncSiteApiRequest(daily, location, listener, pollId);
return true;
}
public void sendAsyncSiteApiRequest(final boolean daily, final PointType location,
final Response.CompleteListener listener, final String pollId) {
final String url = ((daily) ? GET_FORECAST_URL_DAILY : GET_FORECAST_URL_HOURLY)
.replace(GET_FORECAST_KEY_LATITUDE, location.getLatitude().toString())
.replace(GET_FORECAST_KEY_LONGITUDE, location.getLongitude().toString());
final Request request = httpClient.newRequest(url).method(HttpMethod.GET)
.header(HttpHeaders.ACCEPT, MediaType.APPLICATION_JSON_TYPE.toString())
.timeout(GET_FORECAST_REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS);
logger.trace("Requesting using Poll ID \"{}\" URL: \"{}\"", pollId, url);
try {
apiAuth.addAuthentication(request).send(listener);
} catch (AuthTokenException ate) {
notifyAuthenticationListeners(false);
}
}
// Localization functionality
public String getLocalizedText(String key, @Nullable Object @Nullable... arguments) {
String result = translationProvider.getText(bundle, key, key, localeProvider.getLocale(), arguments);
return Objects.nonNull(result) ? result : key;
}
}

View File

@ -0,0 +1,128 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.api;
import static org.openhab.binding.metofficedatahub.internal.MetOfficeDataHubBindingConstants.GET_FORECAST_API_KEY_HEADER;
import static org.openhab.binding.metofficedatahub.internal.MetOfficeDataHubBindingConstants.GSON;
import java.nio.charset.StandardCharsets;
import java.util.Base64;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jetty.client.api.Request;
import org.eclipse.jetty.client.api.Result;
import org.eclipse.jetty.http.HttpStatus;
import com.google.gson.JsonSyntaxException;
/**
* This handles the authentication aspects of the Site API
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public class SiteApiAuthentication {
private final Object isAuthenticatedWriteLock = new Object();
private Boolean isAuthenticated = false;
private String apiKey = "";
public SiteApiAuthentication() {
}
public void dispose() {
}
public void setApiKey(final String newApiKey) throws AuthTokenException {
this.apiKey = "";
// Perform some basic token checks, as data that isn't even JWT formatted will give a
// 500 response from the Met Office servers rather than a 401 response.
final String[] chunks = newApiKey.split("\\.");
if (chunks.length != 3) {
throw new AuthTokenException();
}
final Base64.Decoder decoder = Base64.getUrlDecoder();
try {
final JwtTokenHeader headers = GSON.fromJson(new String(decoder.decode(chunks[0]), StandardCharsets.UTF_8),
JwtTokenHeader.class);
if (headers == null || !headers.isValid()) {
throw new AuthTokenException();
}
final JwtTokenPayload payload = GSON.fromJson(new String(decoder.decode(chunks[1]), StandardCharsets.UTF_8),
JwtTokenPayload.class);
if (payload == null || !payload.isValid()) {
throw new AuthTokenException();
}
decoder.decode(chunks[2]); // check base64 encoding of signature
} catch (JsonSyntaxException | IllegalArgumentException e) {
throw new AuthTokenException();
}
this.apiKey = newApiKey;
}
/**
* Adds the required data to the Jetty Request, for the authentication of the request
*
* @param req is the request to have the relevant headers, etc added
* @return Request is the passed to Request arg to allow chained call's.
*/
public Request addAuthentication(final Request req) throws AuthTokenException {
if (apiKey.isBlank()) {
throw new AuthTokenException();
}
return req.header(GET_FORECAST_API_KEY_HEADER, apiKey);
}
/**
* Process the given Result from a Jetty Client request, and returns true if the Result
* indicates that there was not a authentication issue.
*
* @param result is the Jetty Client Result to check for authentication issues
* @return Result is the passed to Result arg to allow chained call's.
*/
public Result processResult(final Result result) {
if (result.isSucceeded()) {
switch (result.getResponse().getStatus()) {
case HttpStatus.FORBIDDEN_403:
setIsAuthenticated(false);
break;
case HttpStatus.OK_200:
setIsAuthenticated(true);
break;
}
}
return result;
}
/**
* Return whether the current key has been validated as authenticated
*
* @return true if the last result processed indicated a validated api key.
*/
public boolean getIsAuthenticated() {
return this.isAuthenticated;
}
/**
* Set's the state of the isAuthenticated state, using COW principles.
*/
private void setIsAuthenticated(final boolean isAuthenticated) {
synchronized (isAuthenticatedWriteLock) {
this.isAuthenticated = Boolean.valueOf(isAuthenticated);
}
}
}

View File

@ -0,0 +1,44 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.dto.responses;
import com.google.gson.annotations.SerializedName;
/**
* The {@link SiteApiFeature} is a Java class used as a DTO to hold part of the response to the Site Specific API.
*
* @author David Goodyear - Initial contribution
*/
public class SiteApiFeature extends SiteApiTypedResponseObject {
public static final String TYPE_SITE_API_FEATURE = "Feature";
@Override
public String getExpectedType() {
return TYPE_SITE_API_FEATURE;
}
@SerializedName("geometry")
private SiteApiFeaturePoint geometry;
public SiteApiFeaturePoint getGeometry() {
return geometry;
}
@SerializedName("properties")
private SiteApiFeatureProperties properties;
public SiteApiFeatureProperties getProperties() {
return properties;
}
}

View File

@ -0,0 +1,45 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.dto.responses;
import com.google.gson.annotations.SerializedName;
/**
* The {@link SiteApiFeatureCollection} is a Java class used as a DTO to hold the response to the Site Specific API.
*
* @author David Goodyear - Initial contribution
*/
public class SiteApiFeatureCollection extends SiteApiTypedResponseObject {
public static final String TYPE_SITE_API_FEATURE_COLLECTION = "FeatureCollection";
@Override
public String getExpectedType() {
return TYPE_SITE_API_FEATURE_COLLECTION;
}
@SerializedName("features")
private SiteApiFeature[] feature;
public SiteApiFeature[] getFeature() {
return feature;
}
public SiteApiFeatureProperties getFirstProperties() {
if (feature == null || feature.length == 0) {
return null;
}
return feature[0].getProperties();
}
}

View File

@ -0,0 +1,45 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.dto.responses;
import com.google.gson.annotations.SerializedName;
/**
* The {@link SiteApiFeaturePoint} is a Java class used as a DTO to hold part of the response to the Site Specific API.
*
* @author David Goodyear - Initial contribution
*/
public class SiteApiFeaturePoint extends SiteApiTypedResponseObject {
public static final String TYPE_SITE_API_FEATURE = "Point";
@Override
public String getExpectedType() {
return TYPE_SITE_API_FEATURE;
}
@SerializedName("coordinates")
private double[] coordinates;
public double getLongitude() {
return coordinates[0];
}
public double getLatitude() {
return coordinates[1];
}
public double getElevation() {
return coordinates[2];
}
}

View File

@ -0,0 +1,95 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.dto.responses;
import java.util.HashMap;
import com.google.gson.annotations.SerializedName;
/**
* The {@link SiteApiFeatureProperties} is a Java class used as a DTO to hold part of the response to the Site Specific
* API.
*
* @author David Goodyear - Initial contribution
*/
public class SiteApiFeatureProperties {
public class SiteLocation {
@SerializedName("name")
private String name;
public String getName() {
return name;
}
}
@SerializedName("location")
private SiteLocation location;
public SiteLocation getLocation() {
return location;
}
@SerializedName("requestPointDistance")
private double requestPointDistance;
public double getRequestPointDistance() {
return requestPointDistance;
}
@SerializedName("modelRunDate")
private String modelRunDate;
public String getModelRunDate() {
return modelRunDate;
}
@SerializedName("timeSeries")
private SiteApiTimeSeries[] timeSeries;
public SiteApiTimeSeries[] getTimeSeries() {
return timeSeries;
}
public SiteApiTimeSeries getTimeSeries(final int position) {
if (position < 0 || position > timeSeries.length - 1) {
return EMPTY_TIME_SERIES;
} else {
return timeSeries[position];
}
}
public SiteApiTimeSeries getTimeSeries(final String timestamp) {
return getTimeSeries(getHourlyTimeSeriesPositionForCurrentHour(timestamp));
}
public static final SiteApiTimeSeries EMPTY_TIME_SERIES = new SiteApiTimeSeries();
private HashMap<String, Integer> timeseriesPositions = null;
public int getHourlyTimeSeriesPositionForCurrentHour(String timestamp) {
if (timeseriesPositions == null) {
timeseriesPositions = new HashMap<>();
// Populate the lookup table
for (int i = 0; i < timeSeries.length; i++) {
timeseriesPositions.put(timeSeries[i].getTime(), i);
}
}
Integer result = timeseriesPositions.get(timestamp);
if (result == null) {
return -1;
} else {
return result.intValue();
}
}
}

View File

@ -0,0 +1,458 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.dto.responses;
import com.google.gson.annotations.SerializedName;
/**
* The {@link SiteApiTimeSeries} is a Java class used as a DTO to hold part of the response to the Site Specific
* API.
*
* @author David Goodyear - Initial contribution
*/
public class SiteApiTimeSeries {
@SerializedName("time")
private String time;
public String getTime() {
return time;
}
/**
* Fields below relate to the hourly data-model
*/
@SerializedName("screenTemperature")
private Double screenTemperature;
public Double getScreenTemperature() {
return screenTemperature;
}
@SerializedName("maxScreenAirTemp")
private Double maxScreenAirTemp;
public Double getMaxScreenTemperature() {
return maxScreenAirTemp;
}
@SerializedName("minScreenAirTemp")
private Double minScreenAirTemp;
public Double getMinScreenTemperature() {
return minScreenAirTemp;
}
@SerializedName("screenDewPointTemperature")
private Double screenDewPointTemperature;
public Double getScreenDewPointTemperature() {
return screenDewPointTemperature;
}
@SerializedName("feelsLikeTemperature")
private Double feelsLikeTemperature;
public Double getFeelsLikeTemperature() {
return feelsLikeTemperature;
}
@SerializedName("windSpeed10m")
private Double windSpeed10m;
public Double getWindSpeed10m() {
return windSpeed10m;
}
@SerializedName("windDirectionFrom10m")
private Double windDirectionFrom10m;
public Double getWindDirectionFrom10m() {
return windDirectionFrom10m;
}
@SerializedName("max10mWindGust")
private Double max10mWindGust;
public Double getMax10mWindGust() {
return max10mWindGust;
}
@SerializedName("windGustSpeed10m")
private Double windGustSpeed10m;
public Double getWindGustSpeed10m() {
return windGustSpeed10m;
}
@SerializedName("visibility")
private Integer visibility;
public Integer getVisibility() {
return visibility;
}
@SerializedName("screenRelativeHumidity")
private Double screenRelativeHumidity;
public Double getScreenRelativeHumidity() {
return screenRelativeHumidity;
}
@SerializedName("mslp")
private Integer pressure;
public Integer getPressure() {
return pressure;
}
@SerializedName("uvIndex")
private Integer uvIndex;
public Integer getUvIndex() {
return uvIndex;
}
@SerializedName("significantWeatherCode")
private Integer significantWeatherCode;
public Integer getSignificantWeatherCode() {
return significantWeatherCode;
}
@SerializedName("precipitationRate")
private Double precipitationRate;
public Double getPrecipitationRate() {
return precipitationRate;
}
@SerializedName("totalPrecipAmount")
private Double totalPrecipAmount;
public Double getTotalPrecipAmount() {
return totalPrecipAmount;
}
@SerializedName("totalSnowAmount")
private Double totalSnowAmount;
public Double getTotalSnowAmount() {
return totalSnowAmount;
}
@SerializedName("probOfPrecipitation")
private Double probOfPrecipitation;
public Double getProbOfPrecipitation() {
return probOfPrecipitation;
}
/**
* Fields below relate to the daily data-model
*/
@SerializedName("midday10MWindSpeed")
private Double midday10MWindSpeed;
public Double getMidday10MWindSpeed() {
return midday10MWindSpeed;
}
@SerializedName("midnight10MWindSpeed")
private Double midnight10MWindSpeed;
public Double getMidnight10MWindSpeed() {
return midnight10MWindSpeed;
}
@SerializedName("midday10MWindDirection")
private Integer midday10MWindDirection;
public Integer getMidday10MWindDirection() {
return midday10MWindDirection;
}
@SerializedName("midnight10MWindDirection")
private Integer midnight10MWindDirection;
public Integer getMidnight10MWindDirection() {
return midnight10MWindDirection;
}
@SerializedName("midday10MWindGust")
private Double midday10MWindGust;
public Double getMidday10MWindGust() {
return midday10MWindGust;
}
@SerializedName("midnight10MWindGust")
private Double midnight10MWindGust;
public Double getMidnight10MWindGust() {
return midnight10MWindGust;
}
@SerializedName("middayVisibility")
private Integer middayVisibility;
public Integer getMiddayVisibility() {
return middayVisibility;
}
@SerializedName("midnightVisibility")
private Integer midnightVisibility;
public Integer getMidnightVisibility() {
return midnightVisibility;
}
@SerializedName("middayRelativeHumidity")
private Double middayRelativeHumidity;
public Double getMiddayRelativeHumidity() {
return middayRelativeHumidity;
}
@SerializedName("midnightRelativeHumidity")
private Double midnightRelativeHumidity;
public Double getMidnightRelativeHumidity() {
return midnightRelativeHumidity;
}
@SerializedName("middayMslp")
private Integer middayPressure;
public Integer getMiddayPressure() {
return middayPressure;
}
@SerializedName("midnightMslp")
private Integer midnightPressure;
public Integer getMidnightPressure() {
return midnightPressure;
}
@SerializedName("maxUvIndex")
private Integer maxUvIndex;
public Integer getMaxUvIndex() {
return maxUvIndex;
}
@SerializedName("daySignificantWeatherCode")
private Integer daySignificantWeatherCode;
public Integer getDaySignificantWeatherCode() {
return daySignificantWeatherCode;
}
@SerializedName("nightSignificantWeatherCode")
private Integer nightSignificantWeatherCode;
public Integer getNightSignificantWeatherCode() {
return nightSignificantWeatherCode;
}
@SerializedName("dayMaxScreenTemperature")
private Double dayMaxScreenTemperature;
public Double getDayMaxScreenTemperature() {
return dayMaxScreenTemperature;
}
@SerializedName("nightMinScreenTemperature")
private Double nightMinScreenTemperature;
public Double getNightMinScreenTemperature() {
return nightMinScreenTemperature;
}
@SerializedName("dayUpperBoundMaxTemp")
private Double dayUpperBoundMaxTemp;
public Double getDayUpperBoundMaxTemp() {
return dayUpperBoundMaxTemp;
}
@SerializedName("dayUpperBoundMinTemp")
private Double dayUpperBoundMinTemp;
public Double getDayUpperBoundMinTemp() {
return dayUpperBoundMinTemp;
}
@SerializedName("nightUpperBoundMinTemp")
private Double nightUpperBoundMinTemp;
public Double getNightUpperBoundMinTemp() {
return nightUpperBoundMinTemp;
}
@SerializedName("dayLowerBoundMaxTemp")
private Double dayLowerBoundMaxTemp;
public Double getDayLowerBoundMaxTemp() {
return dayLowerBoundMaxTemp;
}
@SerializedName("nightLowerBoundMinTemp")
private Double nightLowerBoundMinTemp;
public Double getNightLowerBoundMinTemp() {
return nightLowerBoundMinTemp;
}
@SerializedName("dayMaxFeelsLikeTemp")
private Double dayMaxFeelsLikeTemp;
public Double getDayMaxFeelsLikeTemp() {
return dayMaxFeelsLikeTemp;
}
@SerializedName("nightMinFeelsLikeTemp")
private Double nightMinFeelsLikeTemp;
public Double getNightMinFeelsLikeTemp() {
return nightMinFeelsLikeTemp;
}
@SerializedName("dayUpperBoundMaxFeelsLikeTemp")
private Double dayUpperBoundMaxFeelsLikeTemp;
public Double getDayUpperBoundMaxFeelsLikeTemp() {
return dayUpperBoundMaxFeelsLikeTemp;
}
@SerializedName("nightUpperBoundMinFeelsLikeTemp")
private Double nightUpperBoundMinFeelsLikeTemp;
public Double getNightUpperBoundMinFeelsLikeTemp() {
return nightUpperBoundMinFeelsLikeTemp;
}
@SerializedName("dayLowerBoundMaxFeelsLikeTemp")
private Double dayLowerBoundMaxFeelsLikeTemp;
public Double getDayLowerBoundMaxFeelsLikeTemp() {
return dayLowerBoundMaxFeelsLikeTemp;
}
@SerializedName("nightLowerBoundMinFeelsLikeTemp")
private Double nightLowerBoundMinFeelsLikeTemp;
public Double getNightLowerBoundMinFeelsLikeTemp() {
return nightLowerBoundMinFeelsLikeTemp;
}
@SerializedName("dayProbabilityOfPrecipitation")
private Double dayProbabilityOfPrecipitation;
public Double getDayProbabilityOfPrecipitation() {
return dayProbabilityOfPrecipitation;
}
@SerializedName("nightProbabilityOfPrecipitation")
private Double nightProbabilityOfPrecipitation;
public Double getNightProbabilityOfPrecipitation() {
return nightProbabilityOfPrecipitation;
}
@SerializedName("dayProbabilityOfSnow")
private Double dayProbabilityOfSnow;
public Double getDayProbabilityOfSnow() {
return dayProbabilityOfSnow;
}
@SerializedName("nightProbabilityOfSnow")
private Double nightProbabilityOfSnow;
public Double getNightProbabilityOfSnow() {
return nightProbabilityOfSnow;
}
@SerializedName("dayProbabilityOfHeavySnow")
private Double dayProbabilityOfHeavySnow;
public Double getDayProbabilityOfHeavySnow() {
return dayProbabilityOfHeavySnow;
}
@SerializedName("nightProbabilityOfHeavySnow")
private Double nightProbabilityOfHeavySnow;
public Double getNightProbabilityOfHeavySnow() {
return nightProbabilityOfHeavySnow;
}
@SerializedName("dayProbabilityOfRain")
private Double dayProbabilityOfRain;
public Double getDayProbabilityOfRain() {
return dayProbabilityOfRain;
}
@SerializedName("nightProbabilityOfRain")
private Double nightProbabilityOfRain;
public Double getNightProbabilityOfRain() {
return nightProbabilityOfRain;
}
@SerializedName("dayProbabilityOfHeavyRain")
private Double dayProbabilityOfHeavyRain;
public Double getDayProbabilityOfHeavyRain() {
return dayProbabilityOfHeavyRain;
}
@SerializedName("nightProbabilityOfHeavyRain")
private Double nightProbabilityOfHeavyRain;
public Double getNightProbabilityOfHeavyRain() {
return nightProbabilityOfHeavyRain;
}
@SerializedName("dayProbabilityOfHail")
private Double dayProbabilityOfHail;
public Double getDayProbabilityOfHail() {
return dayProbabilityOfHail;
}
@SerializedName("nightProbabilityOfHail")
private Double nightProbabilityOfHail;
public Double getNightProbabilityOfHail() {
return nightProbabilityOfHail;
}
@SerializedName("dayProbabilityOfSferics")
private Double dayProbabilityOfSferics;
public Double getDayProbabilityOfSferics() {
return dayProbabilityOfSferics;
}
@SerializedName("nightProbabilityOfSferics")
private Double nightProbabilityOfSferics;
public Double getNightProbabilityOfSferics() {
return nightProbabilityOfSferics;
}
}

View File

@ -0,0 +1,33 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.dto.responses;
import com.google.gson.annotations.SerializedName;
/**
* The {@link SiteApiTypedResponseObject} is a Java class
* used as a base DTO where the response definitions can be
* identified by the type code embedded within the object.
*
* @author David Goodyear - Initial contribution
*/
public abstract class SiteApiTypedResponseObject {
@SerializedName("type")
private String type;
public String getType() {
return type;
}
public abstract String getExpectedType();
}

View File

@ -0,0 +1,12 @@
<?xml version="1.0" encoding="UTF-8"?>
<addon:addon id="metofficedatahub" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:addon="https://openhab.org/schemas/addon/v1.0.0"
xsi:schemaLocation="https://openhab.org/schemas/addon/v1.0.0 https://openhab.org/schemas/addon-1.0.0.xsd">
<type>binding</type>
<name>MetOfficeDataHub Binding</name>
<description>This is the binding for the Met Office DataHub service.</description>
<connection>cloud</connection>
<countries>gb</countries>
</addon:addon>

View File

@ -0,0 +1,319 @@
# add-on
addon.metofficedatahub.name = MetOfficeDataHub Binding
addon.metofficedatahub.description = This is the binding for the Met Office DataHub service.
# thing types
thing-type.metofficedatahub.account.label = MetOffice DataHub Account
thing-type.metofficedatahub.account.description = MetOffice DataHub API Account
thing-type.metofficedatahub.site.label = Forecast Data
thing-type.metofficedatahub.site.description = Site Specific forecast data
thing-type.metofficedatahub.site.group.current-forecast.label = Forecast Current Hour
thing-type.metofficedatahub.site.group.current-forecast.description = This is the weather forecast for the current hour.
thing-type.metofficedatahub.site.group.current-forecast-plus01.label = Forecast +1 Hour
thing-type.metofficedatahub.site.group.current-forecast-plus01.description = This is the weather forecast in 1 hour.
thing-type.metofficedatahub.site.group.current-forecast-plus02.label = Forecast +2 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus02.description = This is the weather forecast in 2 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus03.label = Forecast +3 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus03.description = This is the weather forecast in 3 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus04.label = Forecast +4 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus04.description = This is the weather forecast in 4 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus05.label = Forecast +5 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus05.description = This is the weather forecast in 5 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus06.label = Forecast +6 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus06.description = This is the weather forecast in 6 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus07.label = Forecast +7 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus07.description = This is the weather forecast in 7 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus08.label = Forecast +8 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus08.description = This is the weather forecast in 8 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus09.label = Forecast +9 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus09.description = This is the weather forecast in 9 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus10.label = Forecast +10 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus10.description = This is the weather forecast in 10 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus11.label = Forecast +11 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus11.description = This is the weather forecast in 11 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus12.label = Forecast +12 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus12.description = This is the weather forecast in 12 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus13.label = Forecast +13 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus13.description = This is the weather forecast in 13 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus14.label = Forecast +14 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus14.description = This is the weather forecast in 14 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus15.label = Forecast +15 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus15.description = This is the weather forecast in 15 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus16.label = Forecast +16 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus16.description = This is the weather forecast in 16 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus17.label = Forecast +17 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus17.description = This is the weather forecast in 17 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus18.label = Forecast +18 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus18.description = This is the weather forecast in 18 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus19.label = Forecast +19 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus19.description = This is the weather forecast in 19 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus20.label = Forecast +20 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus20.description = This is the weather forecast in 20 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus21.label = Forecast +21 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus21.description = This is the weather forecast in 21 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus22.label = Forecast +22 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus22.description = This is the weather forecast in 22 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus23.label = Forecast +23 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus23.description = This is the weather forecast in 23 hours.
thing-type.metofficedatahub.site.group.current-forecast-plus24.label = Forecast +24 Hours
thing-type.metofficedatahub.site.group.current-forecast-plus24.description = This is the weather forecast in 24 hours.
thing-type.metofficedatahub.site.group.daily-forecast.label = Forecast Current Day
thing-type.metofficedatahub.site.group.daily-forecast.description = This is the weather forecast for the current day.
thing-type.metofficedatahub.site.group.daily-forecast-plus01.label = Forecast +1 Day
thing-type.metofficedatahub.site.group.daily-forecast-plus01.description = This is the weather forecast in 1 day.
thing-type.metofficedatahub.site.group.daily-forecast-plus02.label = Forecast +2 Days
thing-type.metofficedatahub.site.group.daily-forecast-plus02.description = This is the weather forecast in 2 days.
thing-type.metofficedatahub.site.group.daily-forecast-plus03.label = Forecast +3 Days
thing-type.metofficedatahub.site.group.daily-forecast-plus03.description = This is the weather forecast in 3 days.
thing-type.metofficedatahub.site.group.daily-forecast-plus04.label = Forecast +4 Days
thing-type.metofficedatahub.site.group.daily-forecast-plus04.description = This is the weather forecast in 4 days.
thing-type.metofficedatahub.site.group.daily-forecast-plus05.label = Forecast +5 Days
thing-type.metofficedatahub.site.group.daily-forecast-plus05.description = This is the weather forecast in 5 days.
thing-type.metofficedatahub.site.group.daily-forecast-plus06.label = Forecast +6 Days
thing-type.metofficedatahub.site.group.daily-forecast-plus06.description = This is the weather forecast in 6 days.
# thing types config
thing-type.config.metofficedatahub.account.siteApiKey.label = Site Specific API Key
thing-type.config.metofficedatahub.account.siteApiKey.description = The API Key for the Site Specific subscription in your MET Office Data Hub account
thing-type.config.metofficedatahub.account.siteRateDailyLimit.label = API Daily Limit
thing-type.config.metofficedatahub.account.siteRateDailyLimit.description = A limit to the number of daily site specific API requests
thing-type.config.metofficedatahub.site.dailyForecastPollRate.label = Daily Poll Interval
thing-type.config.metofficedatahub.site.dailyForecastPollRate.description = The default number of hours to wait between retrieving daily forecast data
thing-type.config.metofficedatahub.site.hourlyForecastPollRate.label = Hourly Poll Interval
thing-type.config.metofficedatahub.site.hourlyForecastPollRate.description = The default number of hours to wait between retrieving hourly forecast data
thing-type.config.metofficedatahub.site.location.label = Weather Location
thing-type.config.metofficedatahub.site.location.description = Location of weather forecast in geographical coordinates (latitude/longitude).
# channel group types
channel-group-type.metofficedatahub.site-daily-forecast-grp.label = Daily Forecast
channel-group-type.metofficedatahub.site-daily-forecast-grp.description = Site Specific Location weather forecast - Daily basis
channel-group-type.metofficedatahub.site-hr-forecast-grp.label = Hourly Forecast
channel-group-type.metofficedatahub.site-hr-forecast-grp.description = Site Specific Location weather forecast - Hourly basis
# channel types
channel-type.metofficedatahub.air-temp-current-type.label = Temperature
channel-type.metofficedatahub.air-temp-current-type.description = Air Temperature
channel-type.metofficedatahub.air-temp-max-type.label = Max. Temperature
channel-type.metofficedatahub.air-temp-max-type.description = Maximum Screen Air Temperature Over Previous Hour
channel-type.metofficedatahub.air-temp-min-type.label = Min. Temperature
channel-type.metofficedatahub.air-temp-min-type.description = Minimum Screen Air Temperature Over Previous Hour
channel-type.metofficedatahub.day-prob-heavy-rain-type.label = Day H.Rain Probab.
channel-type.metofficedatahub.day-prob-heavy-rain-type.description = Probability of Heavy Rain During The Day - Heavy rain is defined as >1mm/hr. Daytime is defined as those forecast times that fall between local dawn and dusk.
channel-type.metofficedatahub.dewpoint-type.label = Dew Point
channel-type.metofficedatahub.dewpoint-type.description = Dew Point Temperature
channel-type.metofficedatahub.feels-like-max-day-type.label = Day Feels Like Max.
channel-type.metofficedatahub.feels-like-max-day-type.description = Day Maximum Feels Like Air Temperature - This is the most likely maximum value over the day based on the ensemble spread. This is the temperature it feels like taking into account humidity and wind chill but not radiation. Daytime is defined as those forecast times that fall between local dawn and dusk.
channel-type.metofficedatahub.feels-like-max-lb-day-type.label = Day Max. Feels Like
channel-type.metofficedatahub.feels-like-max-lb-day-type.description = Lower Bound on Day Maximum Feels Like Air Temperature - This is the lower bound for the maximum value over the day based on the ensemble spread. It is actually given by the 2.5 percentile. This means there is a 97.5% probability that the actual figure will be above this lower bound figure. This is the temperature it feels like taking into account humidity and wind chill but not radiation. Daytime is defined as those forecast times that fall between local dawn and dusk.
channel-type.metofficedatahub.feels-like-max-ub-day-type.label = Day(UB) Feels Like Max.
channel-type.metofficedatahub.feels-like-max-ub-day-type.description = Upper Bound on Day Maximum Feels Like Air Temperature - This is the upper bound for the maximum value over the day based on the ensemble spread. It is actually given by the 97.5 percentile. This means there is a 97.5% probability that the actual figure will be below this upper bound figure. This is the temperature it feels like taking into account humidity and wind chill but not radiation. Daytime is defined as those forecast times that fall between local dawn and dusk.
channel-type.metofficedatahub.feels-like-min-lb-night-type.label = Night(LB) Min. Feels Like
channel-type.metofficedatahub.feels-like-min-lb-night-type.description = Lower Bound on Night Minimum Feels Like Air Temperature - This is the lower bound for the minimum value over the night based on the ensemble spread. It is actually given by the 2.5 percentile. This means there is a 97.5% probability that the actual figure will be above this lower bound figure. This is the temperature it feels like taking into account humidity and wind chill but not radiation. Night-time is defined as those forecast times that fall between local dusk and dawn.
channel-type.metofficedatahub.feels-like-min-night-type.label = Night Feels Light Min.
channel-type.metofficedatahub.feels-like-min-night-type.description = Night Minimum Feels Like Air Temperature - This is the most likely minimum value over the night based on the ensemble spread. This is the temperature it feels like taking into account humidity and wind chill but not radiation. Night-time is defined as those forecast times that fall between local dusk and dawn.
channel-type.metofficedatahub.feels-like-min-ub-night-type.label = Night(UB) Feels Like Min.
channel-type.metofficedatahub.feels-like-min-ub-night-type.description = Upper Bound on Night Minimum Feels Like Air Temperature - This is the upper bound for the minimum value over the night based on the ensemble spread. It is actually given by the 97.5 percentile. This means there is a 97.5% probability that the actual figure will be below this upper bound figure. This is the temperature it feels like taking into account humidity and wind chill but not radiation. Night-time is defined as those forecast times that fall between local dusk and dawn.
channel-type.metofficedatahub.feels-like-type.label = Feels Like Temperature
channel-type.metofficedatahub.feels-like-type.description = Feels Like Temperature
channel-type.metofficedatahub.hail-prob-day-type.label = Day Hail Probab.
channel-type.metofficedatahub.hail-prob-day-type.description = Probability of Hail During The Day - Daytime is defined as those forecast times that fall between local dawn and dusk.
channel-type.metofficedatahub.hail-prob-night-type.label = Night Hail Probab.
channel-type.metofficedatahub.hail-prob-night-type.description = Probability of Hail During The Night - Night-time is defined as those forecast times that fall between local dusk and dawn.
channel-type.metofficedatahub.heavy-snow-prob-day-type.label = Day H.Snow Probab.
channel-type.metofficedatahub.heavy-snow-prob-day-type.description = Probability of Heavy Snow During The Day - Heavy snow is defined as >1mm/hr liquid water equivalent and is approximately equivilent to >1cm snow per hour. Daytime is defined as those forecast times that fall between local dawn and dusk.
channel-type.metofficedatahub.heavy-snow-prob-night-type.label = Night H.Snow Probab.
channel-type.metofficedatahub.heavy-snow-prob-night-type.description = Probability of Heavy Snow During The Night - Heavy snow is defined as >1mm/hr liquid water equivalent and is approximately equivilent to >1cm snow per hour. Night-time is defined as those forecast times that fall between local dusk and dawn.
channel-type.metofficedatahub.humidity-day-type.label = Midday Humidity
channel-type.metofficedatahub.humidity-day-type.description = Relative Humidity at Local Midday - Stevenson screen height is approximately 1.5m above ground level.
channel-type.metofficedatahub.humidity-night-type.label = Midnight Humidity
channel-type.metofficedatahub.humidity-night-type.description = Relative Humidity at Local Midnight - Stevenson screen height is approximately 1.5m above ground level.
channel-type.metofficedatahub.humidity-type.label = Relative Humidity
channel-type.metofficedatahub.humidity-type.description = Screen Relative Humidity
channel-type.metofficedatahub.loc-name-type.label = Location Name
channel-type.metofficedatahub.loc-name-type.description = Name of location represented
channel-type.metofficedatahub.night-prob-heavy-rain-type.label = Night H.Rain Probab.
channel-type.metofficedatahub.night-prob-heavy-rain-type.description = Probability of Heavy Rain During The Night - Heavy rain is defined as >1mm/hr. Night-time is defined as those forecast times that fall between local dusk and dawn.
channel-type.metofficedatahub.nightLowerBoundMaxTemp.label = Day(LB) Max. Temperature
channel-type.metofficedatahub.nightLowerBoundMaxTemp.description = Lower Bound on Day Maximum Screen Air Temperature - This is the lower bound for the minimum value over the day based on the ensemble spread. It is actually given by the 2.5 percentile. This means there is a 97.5% probability that the actual figure will be above this lower bound figure. Stevenson screen height is approximately 1.5m above ground level. Night-time is defined as those forecast times that fall between local dusk and dawn.
channel-type.metofficedatahub.precip-prob-day-type.label = Day Precip. Probab.
channel-type.metofficedatahub.precip-prob-day-type.description = Probability of Precipitation During The Day - Daytime is defined as those forecast times that fall between local dawn and dusk.
channel-type.metofficedatahub.precip-prob-night-type.label = Night Precip. Probab.
channel-type.metofficedatahub.precip-prob-night-type.description = Probability of Precipitation During The Night - Night-time is defined as those forecast times that fall between local dusk and dawn.
channel-type.metofficedatahub.precip-prob-type.label = Precip. Probability
channel-type.metofficedatahub.precip-prob-type.description = Probability of Precipitation
channel-type.metofficedatahub.precip-rate-type.label = Precipitation Rate
channel-type.metofficedatahub.precip-rate-type.description = Precipitation Rate
channel-type.metofficedatahub.precip-total-type.label = Previous Hour Precip.
channel-type.metofficedatahub.precip-total-type.description = Total Precipitation Amount Over Previous Hour
channel-type.metofficedatahub.pressure-day-type.label = Midday Pressure
channel-type.metofficedatahub.pressure-day-type.description = Mean Sea Level Pressure at Local Midnight - Air pressure at mean sea level which is close to the geoid in sea areas. Air pressure at sea level is the quantity often abbreviated as pressure or PMSL.
channel-type.metofficedatahub.pressure-night-type.label = Midnight Pressure
channel-type.metofficedatahub.pressure-night-type.description = Mean Sea Level Pressure at Local Midnight - Air pressure at mean sea level which is close to the geoid in sea areas. Air pressure at sea level is the quantity often abbreviated as pressure or PMSL.
channel-type.metofficedatahub.pressure-type.label = Pressure
channel-type.metofficedatahub.pressure-type.description = Mean Sea Level Pressure
channel-type.metofficedatahub.rain-prob-day-type.label = Day Rain Probab.
channel-type.metofficedatahub.rain-prob-day-type.description = Probability of Rain During The Day - Daytime is defined as those forecast times that fall between local dawn and dusk.
channel-type.metofficedatahub.rain-prob-night-type.label = Night Rain Probab.
channel-type.metofficedatahub.rain-prob-night-type.description = Probability of Rain During The Night - Night-time is defined as those forecast times that fall between local dusk and dawn.
channel-type.metofficedatahub.sferics-prob-day-type.label = Day Sferics Probab.
channel-type.metofficedatahub.sferics-prob-day-type.description = Probability of Sferics During The Day - This is the probability of a strike within a radius of 50km.
channel-type.metofficedatahub.sferics-prob-night-type.label = Night Sferics Probab.
channel-type.metofficedatahub.sferics-prob-night-type.description = Probability of Sferics During The Night - This is the probability of a strike within a radius of 50km.
channel-type.metofficedatahub.snow-prob-day-type.label = Day Snow Probab.
channel-type.metofficedatahub.snow-prob-day-type.description = Probability of Snow During The Day - Daytime is defined as those forecast times that fall between local dawn and dusk.
channel-type.metofficedatahub.snow-prob-night-type.label = Night Snow Probab.
channel-type.metofficedatahub.snow-prob-night-type.description = Probability of Snow During The Night - Night-time is defined as those forecast times that fall between local dusk and dawn.
channel-type.metofficedatahub.snow-total-type.label = Previous Hour Snowfall
channel-type.metofficedatahub.snow-total-type.description = Total Snowfall Amount Over Previous Hour
channel-type.metofficedatahub.temp-max-day-type.label = Day Max. Temperature
channel-type.metofficedatahub.temp-max-day-type.description = Day Maximum Screen Air Temperature - Daytime is defined as those forecast times that fall between local dawn and dusk
channel-type.metofficedatahub.temp-max-lb-day-type.label = Day(LB) Max. Temperature
channel-type.metofficedatahub.temp-max-lb-day-type.description = Lower Bound on Day Maximum Screen Air Temperature - This is the lower bound for the maximum value over the day based on the ensemble spread. It is actually given by the 2.5 percentile. This means there is a 97.5% probability that the actual figure will be above this lower bound figure. Stevenson screen height is approximately 1.5m above ground level. Daytime is defined as those forecast times that fall between local dawn and dusk.
channel-type.metofficedatahub.temp-max-ub-day-type.label = Day(UB) Max. Temperature
channel-type.metofficedatahub.temp-max-ub-day-type.description = Upper Bound on Day Maximum Screen Air Temperature - This is the upper bound for the maximum value over the day based on the ensemble spread. It is actually given by the 97.5 percentile. This means there is a 97.5% probability that the actual figure will be below this upper bound figure. Stevenson screen height is approximately 1.5m above ground level. Daytime is defined as those forecast times that fall between local dawn and dusk.
channel-type.metofficedatahub.temp-min-lb-night-type.label = Night(LB) Min. Temperature
channel-type.metofficedatahub.temp-min-lb-night-type.description = Lower Bound on Night Minimum Screen Air Temperature - This is the lower bound for the minimum value over the night based on the ensemble spread. It is actually given by the 2.5 percentile. This means there is a 97.5% probability that the actual figure will be above this lower bound figure. Stevenson screen height is approximately 1.5m above ground level. Night-time is defined as those forecast times that fall between local dusk and dawn.
channel-type.metofficedatahub.temp-min-night-type.label = Night Min. Temperature
channel-type.metofficedatahub.temp-min-night-type.description = Night Minimum Screen Air Temperature - Night-time is defined as those forecast times that fall between local dusk and dawn
channel-type.metofficedatahub.temp-min-ub-night-type.label = Night(UB) Min. Temperature
channel-type.metofficedatahub.temp-min-ub-night-type.description = Upper Bound on Night Minimum Screen Air Temperature - This is the upper bound for the minimum value over the night based on the ensemble spread. It is actually given by the 97.5 percentile. This means there is a 97.5% probability that the actual figure will be below this upper bound figure. Stevenson screen height is approximately 1.5m above ground level. Night-time is defined as those forecast times that fall between local dusk and dawn.
channel-type.metofficedatahub.time-type.label = Forecast Time Start
channel-type.metofficedatahub.time-type.description = Time of forecast time window start
channel-type.metofficedatahub.time-type.state.pattern = %1$tF %1$tR
channel-type.metofficedatahub.uv-index-type.label = UV Index
channel-type.metofficedatahub.uv-index-type.description = UV Index
channel-type.metofficedatahub.uv-max-type.label = Day Max. UV
channel-type.metofficedatahub.uv-max-type.description = Day Maximum UV Index - Usually a value from 0 to 13 but higher values are possible in extreme situations. Daytime is defined as those forecast times that fall between local dawn and dusk.
channel-type.metofficedatahub.visibility-day-type.label = Midday Visibility
channel-type.metofficedatahub.visibility-day-type.description = Visibility at Local Midday - Minimal horizontal distance at which a known object can be seen.
channel-type.metofficedatahub.visibility-night-type.label = Midnight Visibility
channel-type.metofficedatahub.visibility-night-type.description = Visibility at Local Midnight - Minimal horizontal distance at which a known object can be seen.
channel-type.metofficedatahub.visibility-type.label = Visibility
channel-type.metofficedatahub.visibility-type.description = Visibility
channel-type.metofficedatahub.wind-direction-day-type.label = Midday Wind Direction
channel-type.metofficedatahub.wind-direction-day-type.description = 10m Wind Direction at Local Midday - Mean wind direction is equivalent to the mean direction observed over the 10 minutes preceding the validity time. In meteorological reports the direction of the wind vector is given as the direction from which it is blowing. 10m wind is the considered surface wind.
channel-type.metofficedatahub.wind-direction-night-type.label = Midnight Wind Direction
channel-type.metofficedatahub.wind-direction-night-type.description = 10m Wind Direction at Local Midnight - Mean wind direction is equivalent to the mean direction observed over the 10 minutes preceding the validity time. In meteorological reports the direction of the wind vector is given as the direction from which it is blowing. 10m wind is the considered surface wind.
channel-type.metofficedatahub.wind-direction-type.label = Wind From
channel-type.metofficedatahub.wind-direction-type.description = 10m Wind From Direction
channel-type.metofficedatahub.wind-gust-day-type.label = Midday Wind Gust
channel-type.metofficedatahub.wind-gust-day-type.description = 10m Wind Gust Speed at Local Midday - The gust speed is equivalent to the maximum 3 second mean wind speed observed over the 10 minutes preceding the validity time. 10m wind is the considered surface wind.
channel-type.metofficedatahub.wind-gust-max-type.label = Max Wind Gust Prev.Hr
channel-type.metofficedatahub.wind-gust-max-type.description = Maximum 10m Wind Gust Speed of Previous Hour
channel-type.metofficedatahub.wind-gust-night-type.label = Midnight Wind Gust
channel-type.metofficedatahub.wind-gust-night-type.description = 10m Wind Gust Speed at Local Midnight - The gust speed is equivalent to the maximum 3 second mean wind speed observed over the 10 minutes preceding the validity time. 10m wind is the considered surface wind.
channel-type.metofficedatahub.wind-speed-day-type.label = Midday Wind Speed
channel-type.metofficedatahub.wind-speed-day-type.description = 10m Wind Speed at Local Midday - Mean wind speed is equivalent to the mean speed observed over the 10 minutes preceding the validity time. 10m wind is the considered surface wind.
channel-type.metofficedatahub.wind-speed-gust-type.label = Wind Gust
channel-type.metofficedatahub.wind-speed-gust-type.description = 10m Wind Gust Speed
channel-type.metofficedatahub.wind-speed-night-type.label = Midnight Wind Speed
channel-type.metofficedatahub.wind-speed-night-type.description = 10m Wind Speed at Local Midnight - Mean wind speed is equivalent to the mean speed observed over the 10 minutes preceding the validity time. 10m wind is the considered surface wind.
channel-type.metofficedatahub.wind-speed-type.label = Wind Speed
channel-type.metofficedatahub.wind-speed-type.description = 10m Wind Speed
# thing types
thing-type.metofficedatahub.site.group.common-data.label = Common Forecast Data
thing-type.metofficedatahub.site.group.common-data.description = This is common data to all forecast data.
# channel group types
channel-group-type.metofficedatahub.site-common-data-grp.label = Common Site Data
channel-group-type.metofficedatahub.site-common-data-grp.description = Site Specific Location weather forecast - Common Data
# thing types
thing-type.metofficedatahub.siteSpecificApi.label = Forecast Data
thing-type.metofficedatahub.siteSpecificApi.description = Site Specific forecast data
thing-type.metofficedatahub.siteSpecificApi.group.common-data.label = Common Forecast Data
thing-type.metofficedatahub.siteSpecificApi.group.common-data.description = This is common data to all forecast data.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast.label = Forecast Current Hour
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast.description = This is the weather forecast for the current hour.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus01.label = Forecast +1 Hour
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus01.description = This is the weather forecast in 1 hour.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus02.label = Forecast +2 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus02.description = This is the weather forecast in 2 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus03.label = Forecast +3 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus03.description = This is the weather forecast in 3 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus04.label = Forecast +4 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus04.description = This is the weather forecast in 4 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus05.label = Forecast +5 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus05.description = This is the weather forecast in 5 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus06.label = Forecast +6 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus06.description = This is the weather forecast in 6 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus07.label = Forecast +7 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus07.description = This is the weather forecast in 7 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus08.label = Forecast +8 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus08.description = This is the weather forecast in 8 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus09.label = Forecast +9 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus09.description = This is the weather forecast in 9 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus10.label = Forecast +10 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus10.description = This is the weather forecast in 10 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus11.label = Forecast +11 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus11.description = This is the weather forecast in 11 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus12.label = Forecast +12 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus12.description = This is the weather forecast in 12 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus13.label = Forecast +13 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus13.description = This is the weather forecast in 13 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus14.label = Forecast +14 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus14.description = This is the weather forecast in 14 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus15.label = Forecast +15 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus15.description = This is the weather forecast in 15 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus16.label = Forecast +16 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus16.description = This is the weather forecast in 16 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus17.label = Forecast +17 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus17.description = This is the weather forecast in 17 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus18.label = Forecast +18 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus18.description = This is the weather forecast in 18 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus19.label = Forecast +19 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus19.description = This is the weather forecast in 19 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus20.label = Forecast +20 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus20.description = This is the weather forecast in 20 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus21.label = Forecast +21 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus21.description = This is the weather forecast in 21 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus22.label = Forecast +22 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus22.description = This is the weather forecast in 22 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus23.label = Forecast +23 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus23.description = This is the weather forecast in 23 hours.
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus24.label = Forecast +24 Hours
thing-type.metofficedatahub.siteSpecificApi.group.current-forecast-plus24.description = This is the weather forecast in 24 hours.
thing-type.metofficedatahub.siteSpecificApi.group.daily-forecast.label = Forecast Current Day
thing-type.metofficedatahub.siteSpecificApi.group.daily-forecast.description = This is the weather forecast for the current day.
thing-type.metofficedatahub.siteSpecificApi.group.daily-forecast-plus01.label = Forecast +1 Day
thing-type.metofficedatahub.siteSpecificApi.group.daily-forecast-plus01.description = This is the weather forecast in 1 day.
thing-type.metofficedatahub.siteSpecificApi.group.daily-forecast-plus02.label = Forecast +2 Days
thing-type.metofficedatahub.siteSpecificApi.group.daily-forecast-plus02.description = This is the weather forecast in 2 days.
thing-type.metofficedatahub.siteSpecificApi.group.daily-forecast-plus03.label = Forecast +3 Days
thing-type.metofficedatahub.siteSpecificApi.group.daily-forecast-plus03.description = This is the weather forecast in 3 days.
thing-type.metofficedatahub.siteSpecificApi.group.daily-forecast-plus04.label = Forecast +4 Days
thing-type.metofficedatahub.siteSpecificApi.group.daily-forecast-plus04.description = This is the weather forecast in 4 days.
thing-type.metofficedatahub.siteSpecificApi.group.daily-forecast-plus05.label = Forecast +5 Days
thing-type.metofficedatahub.siteSpecificApi.group.daily-forecast-plus05.description = This is the weather forecast in 5 days.
thing-type.metofficedatahub.siteSpecificApi.group.daily-forecast-plus06.label = Forecast +6 Days
thing-type.metofficedatahub.siteSpecificApi.group.daily-forecast-plus06.description = This is the weather forecast in 6 days.
# thing types config
thing-type.config.metofficedatahub.siteSpecificApi.dailyForecastPollRate.label = Daily Poll Cycle
thing-type.config.metofficedatahub.siteSpecificApi.dailyForecastPollRate.description = The default number of hours to wait between retrieving daily forecast data
thing-type.config.metofficedatahub.siteSpecificApi.hourlyForecastPollRate.label = Hourly Poll Cycle
thing-type.config.metofficedatahub.siteSpecificApi.hourlyForecastPollRate.description = The default number of hours to wait between retrieving hourly forecast data
thing-type.config.metofficedatahub.siteSpecificApi.location.label = Weather Location
thing-type.config.metofficedatahub.siteSpecificApi.location.description = Location of weather forecast in geographical coordinates (latitude/longitude).
# channel types
api.log.rate-limiter.failed-restore = Limiter {0} -> Did not load limiter data due to : {1}
bridge.error.site-specific.auth-issue = Check siteSpecificApiKey is correct in account - authentication failure
bridge.error.site-specific.communication-failure = Communications failure due to : {0}
bridge.error.site-specific.communication-failure.unknown = unknown reason
site.error.no-bridge = Disabled requesting data - this things Bridge is not set
site.error.no-user-location = Missing openHAB's user configured location
site.error.invalid-location = Invalid location given
site.error.not-enough-data = No data cached or retrieved recently enough
comm.comm-check.no-quota-left = Request limit reached - cannot check connectivity to Met Office - presuming online

View File

@ -0,0 +1,615 @@
<?xml version="1.0" encoding="UTF-8"?>
<thing:thing-descriptions bindingId="metofficedatahub"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:thing="https://openhab.org/schemas/thing-description/v1.0.0"
xsi:schemaLocation="https://openhab.org/schemas/thing-description/v1.0.0 https://openhab.org/schemas/thing-description-1.0.0.xsd">
<!-- Channel Groups for MetOfficeDataHub Binding - Site API -->
<channel-group-type id="site-hr-forecast-grp">
<label>Hourly Forecast</label>
<description>Site Specific Location weather forecast - Hourly basis</description>
<channels>
<channel id="forecast-ts" typeId="time-type"/>
<channel id="air-temp-current" typeId="air-temp-current-type"/>
<channel id="air-temp-min" typeId="air-temp-min-type"/>
<channel id="air-temp-max" typeId="air-temp-max-type"/>
<channel id="feels-like" typeId="feels-like-type"/>
<channel id="humidity" typeId="humidity-type"/>
<channel id="visibility" typeId="visibility-type"/>
<channel id="precip-prob" typeId="precip-prob-type"/>
<channel id="precip-rate" typeId="precip-rate-type"/>
<channel id="precip-total" typeId="precip-total-type"/>
<channel id="snow-total" typeId="snow-total-type"/>
<channel id="uv-index" typeId="uv-index-type"/>
<channel id="pressure" typeId="pressure-type"/>
<channel id="wind-speed" typeId="wind-speed-type"/>
<channel id="wind-speed-gust" typeId="wind-speed-gust-type"/>
<channel id="wind-gust-max" typeId="wind-gust-max-type"/>
<channel id="wind-direction" typeId="wind-direction-type"/>
<channel id="dewpoint" typeId="dewpoint-type"/>
</channels>
</channel-group-type>
<channel-group-type id="site-daily-forecast-grp">
<label>Daily Forecast</label>
<description>Site Specific Location weather forecast - Daily basis</description>
<channels>
<channel id="forecast-ts" typeId="time-type"/>
<channel id="wind-speed-day" typeId="wind-speed-day-type"/>
<channel id="wind-speed-night" typeId="wind-speed-night-type"/>
<channel id="wind-direction-day" typeId="wind-direction-day-type"/>
<channel id="wind-direction-night" typeId="wind-direction-night-type"/>
<channel id="wind-gust-day" typeId="wind-gust-day-type"/>
<channel id="wind-gust-night" typeId="wind-gust-night-type"/>
<channel id="visibility-day" typeId="visibility-day-type"/>
<channel id="visibility-night" typeId="visibility-night-type"/>
<channel id="humidity-day" typeId="humidity-day-type"/>
<channel id="humidity-night" typeId="humidity-night-type"/>
<channel id="pressure-day" typeId="pressure-day-type"/>
<channel id="pressure-night" typeId="pressure-night-type"/>
<channel id="uv-max" typeId="uv-max-type"/>
<channel id="temp-max-ub-day" typeId="temp-max-ub-day-type"/>
<channel id="temp-min-ub-night" typeId="temp-min-ub-night-type"/>
<channel id="temp-max-lb-day" typeId="temp-max-lb-day-type"/>
<channel id="temp-min-lb-night" typeId="temp-min-lb-night-type"/>
<channel id="feels-like-max-day" typeId="feels-like-max-day-type"/>
<channel id="feels-like-min-night" typeId="feels-like-min-night-type"/>
<channel id="temp-max-day" typeId="temp-max-day-type"/>
<channel id="temp-min-night" typeId="temp-min-night-type"/>
<channel id="feels-like-max-ub-day" typeId="feels-like-max-ub-day-type"/>
<channel id="feels-like-min-ub-night" typeId="feels-like-min-ub-night-type"/>
<channel id="feels-like-max-lb-day" typeId="feels-like-max-lb-day-type"/>
<channel id="feels-like-min-lb-night" typeId="feels-like-min-lb-night-type"/>
<channel id="precip-prob-day" typeId="precip-prob-day-type"/>
<channel id="precip-prob-night" typeId="precip-prob-night-type"/>
<channel id="snow-prob-day" typeId="snow-prob-day-type"/>
<channel id="snow-prob-night" typeId="snow-prob-night-type"/>
<channel id="heavy-snow-prob-day" typeId="heavy-snow-prob-day-type"/>
<channel id="heavy-snow-prob-night" typeId="heavy-snow-prob-night-type"/>
<channel id="rain-prob-day" typeId="rain-prob-day-type"/>
<channel id="rain-prob-night" typeId="rain-prob-night-type"/>
<channel id="day-prob-heavy-rain" typeId="day-prob-heavy-rain-type"/>
<channel id="night-prob-heavy-rain" typeId="night-prob-heavy-rain-type"/>
<channel id="hail-prob-day" typeId="hail-prob-day-type"/>
<channel id="hail-prob-night" typeId="hail-prob-night-type"/>
<channel id="sferics-prob-day" typeId="sferics-prob-day-type"/>
<channel id="sferics-prob-night" typeId="sferics-prob-night-type"/>
</channels>
</channel-group-type>
<!-- Channels for MetOfficeDataHub Binding - Hourly Forecast -->
<channel-type id="loc-name-type">
<item-type>String</item-type>
<label>Location Name</label>
<description>Name of location represented</description>
<state readOnly="true"/>
</channel-type>
<channel-type id="time-type">
<item-type>DateTime</item-type>
<label>Forecast Time Start</label>
<description>Time of forecast time window start</description>
<category>Time</category>
<state readOnly="true" pattern="%1$tF %1$tR"/>
</channel-type>
<channel-type id="air-temp-current-type">
<item-type>Number:Temperature</item-type>
<label>Temperature</label>
<description>Air Temperature</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="air-temp-min-type">
<item-type>Number:Temperature</item-type>
<label>Min. Temperature</label>
<description>Minimum Screen Air Temperature Over Previous Hour</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="air-temp-max-type">
<item-type>Number:Temperature</item-type>
<label>Max. Temperature</label>
<description>Maximum Screen Air Temperature Over Previous Hour</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="feels-like-type">
<item-type>Number:Temperature</item-type>
<label>Feels Like Temperature</label>
<description>Feels Like Temperature</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="humidity-type">
<item-type>Number:Dimensionless</item-type>
<label>Relative Humidity</label>
<description>Screen Relative Humidity</description>
<category>Humidity</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="visibility-type">
<item-type>Number:Length</item-type>
<label>Visibility</label>
<description>Visibility</description>
<category>Sun_Clouds</category>
<state readOnly="true" pattern="%.2f %unit%"/>
</channel-type>
<channel-type id="precip-prob-type">
<item-type>Number:Dimensionless</item-type>
<label>Precip. Probability</label>
<description>Probability of Precipitation</description>
<category>Rain</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="precip-rate-type">
<item-type>Number:Speed</item-type>
<label>Precipitation Rate</label>
<description>Precipitation Rate</description>
<category>Rain</category>
<state readOnly="true" pattern="%.4f %unit%"/>
</channel-type>
<channel-type id="precip-total-type">
<item-type>Number:Length</item-type>
<label>Previous Hour Precip.</label>
<description>Total Precipitation Amount Over Previous Hour</description>
<category>Rain</category>
<state readOnly="true" pattern="%.4f %unit%"/>
</channel-type>
<channel-type id="snow-total-type">
<item-type>Number:Length</item-type>
<label>Previous Hour Snowfall</label>
<description>Total Snowfall Amount Over Previous Hour</description>
<category>Snow</category>
<state readOnly="true" pattern="%.4f %unit%"/>
</channel-type>
<channel-type id="uv-index-type">
<item-type>Number:Dimensionless</item-type>
<label>UV Index</label>
<description>UV Index</description>
<category>Sun</category>
<state readOnly="true" pattern="%.0f"/>
</channel-type>
<channel-type id="pressure-type">
<item-type>Number:Pressure</item-type>
<label>Pressure</label>
<description>Mean Sea Level Pressure</description>
<category>Pressure</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="wind-speed-type">
<item-type>Number:Speed</item-type>
<label>Wind Speed</label>
<description>10m Wind Speed</description>
<category>Wind</category>
<state readOnly="true" pattern="%.2f %unit%"/>
</channel-type>
<channel-type id="wind-speed-gust-type">
<item-type>Number:Speed</item-type>
<label>Wind Gust</label>
<description>10m Wind Gust Speed</description>
<category>Wind</category>
<state readOnly="true" pattern="%.2f %unit%"/>
</channel-type>
<channel-type id="wind-gust-max-type">
<item-type>Number:Speed</item-type>
<label>Max Wind Gust Prev.Hr</label>
<description>Maximum 10m Wind Gust Speed of Previous Hour</description>
<category>Wind</category>
<state readOnly="true" pattern="%.2f %unit%"/>
</channel-type>
<channel-type id="dewpoint-type">
<item-type>Number:Temperature</item-type>
<label>Dew Point</label>
<description>Dew Point Temperature</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="wind-direction-type">
<item-type>Number:Angle</item-type>
<label>Wind From</label>
<description>10m Wind From Direction</description>
<category>Wind</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<!-- Channels for MetOfficeDataHub Binding - Daily Forecast -->
<channel-type id="wind-speed-day-type">
<item-type>Number:Speed</item-type>
<label>Midday Wind Speed</label>
<description>10m Wind Speed at Local Midday - Mean wind speed is equivalent to the mean speed observed over the 10
minutes preceding the validity time. 10m wind is the considered surface wind.</description>
<category>Wind</category>
<state readOnly="true" pattern="%.2f %unit%"/>
</channel-type>
<channel-type id="wind-speed-night-type">
<item-type>Number:Speed</item-type>
<label>Midnight Wind Speed</label>
<description>10m Wind Speed at Local Midnight - Mean wind speed is equivalent to the mean speed observed over the 10
minutes preceding the validity time. 10m wind is the considered surface wind.</description>
<category>Wind</category>
<state readOnly="true" pattern="%.2f %unit%"/>
</channel-type>
<channel-type id="wind-direction-day-type">
<item-type>Number:Angle</item-type>
<label>Midday Wind Direction</label>
<description>10m Wind Direction at Local Midday - Mean wind direction is equivalent to the mean direction observed
over the 10 minutes preceding the validity time. In meteorological reports the direction of the wind vector is given
as the direction from which it is blowing. 10m wind is the considered surface wind.</description>
<category>Wind</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="wind-direction-night-type">
<item-type>Number:Angle</item-type>
<label>Midnight Wind Direction</label>
<description>10m Wind Direction at Local Midnight - Mean wind direction is equivalent to the mean direction observed
over the 10 minutes preceding the validity time. In meteorological reports the direction of the wind vector is given
as the direction from which it is blowing. 10m wind is the considered surface wind.</description>
<category>Wind</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="wind-gust-day-type">
<item-type>Number:Speed</item-type>
<label>Midday Wind Gust</label>
<description>10m Wind Gust Speed at Local Midday - The gust speed is equivalent to the maximum 3 second mean wind
speed observed over the 10 minutes preceding the validity time. 10m wind is the considered surface wind.</description>
<category>Wind</category>
<state readOnly="true" pattern="%.2f %unit%"/>
</channel-type>
<channel-type id="wind-gust-night-type">
<item-type>Number:Speed</item-type>
<label>Midnight Wind Gust</label>
<description>10m Wind Gust Speed at Local Midnight - The gust speed is equivalent to the maximum 3 second mean wind
speed observed over the 10 minutes preceding the validity time. 10m wind is the considered surface wind.</description>
<category>Wind</category>
<state readOnly="true" pattern="%.2f %unit%"/>
</channel-type>
<channel-type id="visibility-day-type">
<item-type>Number:Length</item-type>
<label>Midday Visibility</label>
<description>Visibility at Local Midday - Minimal horizontal distance at which a known object can be seen.</description>
<category>Sun_Clouds</category>
<state readOnly="true" pattern="%.2f %unit%"/>
</channel-type>
<channel-type id="visibility-night-type">
<item-type>Number:Length</item-type>
<label>Midnight Visibility</label>
<description>Visibility at Local Midnight - Minimal horizontal distance at which a known object can be seen.</description>
<category>Sun_Clouds</category>
<state readOnly="true" pattern="%.2f %unit%"/>
</channel-type>
<channel-type id="humidity-day-type">
<item-type>Number:Dimensionless</item-type>
<label>Midday Humidity</label>
<description>Relative Humidity at Local Midday - Stevenson screen height is approximately 1.5m above ground level.</description>
<category>Humidity</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="humidity-night-type">
<item-type>Number:Dimensionless</item-type>
<label>Midnight Humidity</label>
<description>Relative Humidity at Local Midnight - Stevenson screen height is approximately 1.5m above ground level.</description>
<category>Humidity</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="pressure-day-type">
<item-type>Number:Pressure</item-type>
<label>Midday Pressure</label>
<description>Mean Sea Level Pressure at Local Midnight - Air pressure at mean sea level which is close to the geoid in
sea areas. Air pressure at sea level is the quantity often abbreviated as pressure or PMSL.</description>
<category>Pressure</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="pressure-night-type">
<item-type>Number:Pressure</item-type>
<label>Midnight Pressure</label>
<description>Mean Sea Level Pressure at Local Midnight - Air pressure at mean sea level which is close to the geoid in
sea areas. Air pressure at sea level is the quantity often abbreviated as pressure or PMSL.</description>
<category>Pressure</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="uv-max-type">
<item-type>Number:Dimensionless</item-type>
<label>Day Max. UV</label>
<description>Day Maximum UV Index - Usually a value from 0 to 13 but higher values are possible in extreme situations.
Daytime is defined as those forecast times that fall between local dawn and dusk.</description>
<category>Sun</category>
<state readOnly="true" pattern="%.0f"/>
</channel-type>
<channel-type id="temp-max-ub-day-type">
<item-type>Number:Temperature</item-type>
<label>Day(UB) Max. Temperature</label>
<description>Upper Bound on Day Maximum Screen Air Temperature - This is the upper bound for the maximum value over
the day based on the ensemble spread. It is actually given by the 97.5 percentile. This means there is a 97.5%
probability that the actual figure will be below this upper bound figure. Stevenson screen height is approximately
1.5m above ground level. Daytime is defined as those forecast times that fall between local dawn and dusk.</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="temp-min-ub-night-type">
<item-type>Number:Temperature</item-type>
<label>Night(UB) Min. Temperature</label>
<description>Upper Bound on Night Minimum Screen Air Temperature - This is the upper bound for the minimum value over
the night based on the ensemble spread. It is actually given by the 97.5 percentile. This means there is a 97.5%
probability that the actual figure will be below this upper bound figure. Stevenson screen height is approximately
1.5m above ground level. Night-time is defined as those forecast times that fall between local dusk and dawn.</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="temp-max-lb-day-type">
<item-type>Number:Temperature</item-type>
<label>Day(LB) Max. Temperature</label>
<description>Lower Bound on Day Maximum Screen Air Temperature - This is the lower bound for the maximum value over
the day based on the ensemble spread. It is actually given by the 2.5 percentile. This means there is a 97.5%
probability that the actual figure will be above this lower bound figure. Stevenson screen height is approximately
1.5m above ground level. Daytime is defined as those forecast times that fall between local dawn and dusk.</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="nightLowerBoundMaxTemp">
<item-type>Number:Temperature</item-type>
<label>Day(LB) Max. Temperature</label>
<description>Lower Bound on Day Maximum Screen Air Temperature - This is the lower bound for the minimum value over
the day based on the ensemble spread. It is actually given by the 2.5 percentile. This means there is a 97.5%
probability that the actual figure will be above this lower bound figure. Stevenson screen height is approximately
1.5m above ground level. Night-time is defined as those forecast times that fall between local dusk and dawn.</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="temp-min-lb-night-type">
<item-type>Number:Temperature</item-type>
<label>Night(LB) Min. Temperature</label>
<description>Lower Bound on Night Minimum Screen Air Temperature - This is the lower bound for the minimum value over
the night based on the ensemble spread. It is actually given by the 2.5 percentile. This means there is a 97.5%
probability that the actual figure will be above this lower bound figure. Stevenson screen height is approximately
1.5m above ground level. Night-time is defined as those forecast times that fall between local dusk and dawn.</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="feels-like-max-day-type">
<item-type>Number:Temperature</item-type>
<label>Day Feels Like Max.</label>
<description>Day Maximum Feels Like Air Temperature - This is the most likely maximum value over the day based on the
ensemble spread. This is the temperature it feels like taking into account humidity and wind chill but not radiation.
Daytime is defined as those forecast times that fall between local dawn and dusk.</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="feels-like-min-night-type">
<item-type>Number:Temperature</item-type>
<label>Night Feels Light Min.</label>
<description>Night Minimum Feels Like Air Temperature - This is the most likely minimum value over the night based on
the ensemble spread. This is the temperature it feels like taking into account humidity and wind chill but not
radiation. Night-time is defined as those forecast times that fall between local dusk and dawn.</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="feels-like-max-ub-day-type">
<item-type>Number:Temperature</item-type>
<label>Day(UB) Feels Like Max.</label>
<description>Upper Bound on Day Maximum Feels Like Air Temperature - This is the upper bound for the maximum value
over the day based on the ensemble spread. It is actually given by the 97.5 percentile. This means there is a 97.5%
probability that the actual figure will be below this upper bound figure. This is the temperature it feels like
taking into account humidity and wind chill but not radiation. Daytime is defined as those forecast times that fall
between local dawn and dusk.</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="feels-like-min-ub-night-type">
<item-type>Number:Temperature</item-type>
<label>Night(UB) Feels Like Min.</label>
<description>Upper Bound on Night Minimum Feels Like Air Temperature - This is the upper bound for the minimum value
over the night based on the ensemble spread. It is actually given by the 97.5 percentile. This means there is a 97.5%
probability that the actual figure will be below this upper bound figure. This is the temperature it feels like
taking into account humidity and wind chill but not radiation. Night-time is defined as those forecast times that
fall between local dusk and dawn.</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="feels-like-max-lb-day-type">
<item-type>Number:Temperature</item-type>
<label>Day Max. Feels Like</label>
<description>Lower Bound on Day Maximum Feels Like Air Temperature - This is the lower bound for the maximum value
over the day based on the ensemble spread. It is actually given by the 2.5 percentile. This means there is a 97.5%
probability that the actual figure will be above this lower bound figure. This is the temperature it feels like
taking into account humidity and wind chill but not radiation. Daytime is defined as those forecast times that fall
between local dawn and dusk.</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="feels-like-min-lb-night-type">
<item-type>Number:Temperature</item-type>
<label>Night(LB) Min. Feels Like</label>
<description>Lower Bound on Night Minimum Feels Like Air Temperature - This is the lower bound for the minimum value
over the night based on the ensemble spread. It is actually given by the 2.5 percentile. This means there is a 97.5%
probability that the actual figure will be above this lower bound figure. This is the temperature it feels like
taking into account humidity and wind chill but not radiation. Night-time is defined as those forecast times that
fall between local dusk and dawn.</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="precip-prob-day-type">
<item-type>Number:Dimensionless</item-type>
<label>Day Precip. Probab.</label>
<description>Probability of Precipitation During The Day - Daytime is defined as those forecast times that fall
between local dawn and dusk.</description>
<category>Rain</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="precip-prob-night-type">
<item-type>Number:Dimensionless</item-type>
<label>Night Precip. Probab.</label>
<description>Probability of Precipitation During The Night - Night-time is defined as those forecast times that fall
between local dusk and dawn.</description>
<category>Rain</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="snow-prob-day-type">
<item-type>Number:Dimensionless</item-type>
<label>Day Snow Probab.</label>
<description>Probability of Snow During The Day - Daytime is defined as those forecast times that fall between local
dawn and dusk.</description>
<category>Rain</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="snow-prob-night-type">
<item-type>Number:Dimensionless</item-type>
<label>Night Snow Probab.</label>
<description>Probability of Snow During The Night - Night-time is defined as those forecast times that fall between
local dusk and dawn.</description>
<category>Rain</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="heavy-snow-prob-day-type">
<item-type>Number:Dimensionless</item-type>
<label>Day H.Snow Probab.</label>
<description>Probability of Heavy Snow During The Day - Heavy snow is defined as &gt;1mm/hr liquid water equivalent
and is approximately equivilent to &gt;1cm snow per hour. Daytime is defined as those forecast times that fall
between local dawn and dusk.</description>
<category>Rain</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="heavy-snow-prob-night-type">
<item-type>Number:Dimensionless</item-type>
<label>Night H.Snow Probab.</label>
<description>Probability of Heavy Snow During The Night - Heavy snow is defined as &gt;1mm/hr liquid water equivalent
and is approximately equivilent to &gt;1cm snow per hour. Night-time is defined as those forecast times that fall
between local dusk and dawn.</description>
<category>Rain</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="rain-prob-day-type">
<item-type>Number:Dimensionless</item-type>
<label>Day Rain Probab.</label>
<description>Probability of Rain During The Day - Daytime is defined as those forecast times that fall between local
dawn and dusk.</description>
<category>Rain</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="rain-prob-night-type">
<item-type>Number:Dimensionless</item-type>
<label>Night Rain Probab.</label>
<description>Probability of Rain During The Night - Night-time is defined as those forecast times that fall between
local dusk and dawn.</description>
<category>Rain</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="day-prob-heavy-rain-type">
<item-type>Number:Dimensionless</item-type>
<label>Day H.Rain Probab.</label>
<description>Probability of Heavy Rain During The Day - Heavy rain is defined as &gt;1mm/hr. Daytime is defined as
those forecast times that fall between local dawn and dusk.</description>
<category>Rain</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="night-prob-heavy-rain-type">
<item-type>Number:Dimensionless</item-type>
<label>Night H.Rain Probab.</label>
<description>Probability of Heavy Rain During The Night - Heavy rain is defined as &gt;1mm/hr. Night-time is defined
as those forecast times that fall between local dusk and dawn.</description>
<category>Rain</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="hail-prob-day-type">
<item-type>Number:Dimensionless</item-type>
<label>Day Hail Probab.</label>
<description>Probability of Hail During The Day - Daytime is defined as those forecast times that fall between local
dawn and dusk.</description>
<category>Rain</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="hail-prob-night-type">
<item-type>Number:Dimensionless</item-type>
<label>Night Hail Probab.</label>
<description>Probability of Hail During The Night - Night-time is defined as those forecast times that fall between
local dusk and dawn.</description>
<category>Rain</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="sferics-prob-day-type">
<item-type>Number:Dimensionless</item-type>
<label>Day Sferics Probab.</label>
<description>Probability of Sferics During The Day - This is the probability of a strike within a radius of 50km.</description>
<category>Rain</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="sferics-prob-night-type">
<item-type>Number:Dimensionless</item-type>
<label>Night Sferics Probab.</label>
<description>Probability of Sferics During The Night - This is the probability of a strike within a radius of 50km.</description>
<category>Rain</category>
<state readOnly="true" pattern="%.0f %%"/>
</channel-type>
<channel-type id="temp-max-day-type">
<item-type>Number:Temperature</item-type>
<label>Day Max. Temperature</label>
<description>Day Maximum Screen Air Temperature - Daytime is defined as those forecast times that fall between local
dawn and dusk</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
<channel-type id="temp-min-night-type">
<item-type>Number:Temperature</item-type>
<label>Night Min. Temperature</label>
<description>Night Minimum Screen Air Temperature - Night-time is defined as those forecast times that fall between
local dusk and dawn</description>
<category>Temperature</category>
<state readOnly="true" pattern="%.1f %unit%"/>
</channel-type>
</thing:thing-descriptions>

View File

@ -0,0 +1,193 @@
<?xml version="1.0" encoding="UTF-8"?>
<thing:thing-descriptions bindingId="metofficedatahub"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:thing="https://openhab.org/schemas/thing-description/v1.0.0"
xsi:schemaLocation="https://openhab.org/schemas/thing-description/v1.0.0 https://openhab.org/schemas/thing-description-1.0.0.xsd">
<bridge-type id="account">
<label>MetOffice DataHub Account</label>
<description>MetOffice DataHub API Account</description>
<properties>
<property name="Site Specific API Call Count"/>
</properties>
<config-description>
<parameter name="siteApiKey" type="text" required="true">
<context>password</context>
<label>Site Specific API Key</label>
<description>The API Key for the Site Specific subscription in your MET Office Data Hub account</description>
</parameter>
<parameter name="siteRateDailyLimit" type="integer" unit="s" min="1">
<label>API Daily Limit</label>
<description>A limit to the number of daily site specific API requests</description>
<default>250</default>
<advanced>true</advanced>
</parameter>
</config-description>
</bridge-type>
<thing-type id="site">
<supported-bridge-type-refs>
<bridge-type-ref id="account"/>
</supported-bridge-type-refs>
<label>Forecast Data</label>
<description>Site Specific forecast data</description>
<channel-groups>
<channel-group id="current-forecast" typeId="site-hr-forecast-grp">
<label>Forecast Current Hour</label>
<description>This is the weather forecast for the current hour.</description>
</channel-group>
<channel-group id="current-forecast-plus01" typeId="site-hr-forecast-grp">
<label>Forecast +1 Hour</label>
<description>This is the weather forecast in 1 hour.</description>
</channel-group>
<channel-group id="current-forecast-plus02" typeId="site-hr-forecast-grp">
<label>Forecast +2 Hours</label>
<description>This is the weather forecast in 2 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus03" typeId="site-hr-forecast-grp">
<label>Forecast +3 Hours</label>
<description>This is the weather forecast in 3 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus04" typeId="site-hr-forecast-grp">
<label>Forecast +4 Hours</label>
<description>This is the weather forecast in 4 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus05" typeId="site-hr-forecast-grp">
<label>Forecast +5 Hours</label>
<description>This is the weather forecast in 5 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus06" typeId="site-hr-forecast-grp">
<label>Forecast +6 Hours</label>
<description>This is the weather forecast in 6 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus07" typeId="site-hr-forecast-grp">
<label>Forecast +7 Hours</label>
<description>This is the weather forecast in 7 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus08" typeId="site-hr-forecast-grp">
<label>Forecast +8 Hours</label>
<description>This is the weather forecast in 8 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus09" typeId="site-hr-forecast-grp">
<label>Forecast +9 Hours</label>
<description>This is the weather forecast in 9 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus10" typeId="site-hr-forecast-grp">
<label>Forecast +10 Hours</label>
<description>This is the weather forecast in 10 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus11" typeId="site-hr-forecast-grp">
<label>Forecast +11 Hours</label>
<description>This is the weather forecast in 11 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus12" typeId="site-hr-forecast-grp">
<label>Forecast +12 Hours</label>
<description>This is the weather forecast in 12 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus13" typeId="site-hr-forecast-grp">
<label>Forecast +13 Hours</label>
<description>This is the weather forecast in 13 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus14" typeId="site-hr-forecast-grp">
<label>Forecast +14 Hours</label>
<description>This is the weather forecast in 14 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus15" typeId="site-hr-forecast-grp">
<label>Forecast +15 Hours</label>
<description>This is the weather forecast in 15 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus16" typeId="site-hr-forecast-grp">
<label>Forecast +16 Hours</label>
<description>This is the weather forecast in 16 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus17" typeId="site-hr-forecast-grp">
<label>Forecast +17 Hours</label>
<description>This is the weather forecast in 17 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus18" typeId="site-hr-forecast-grp">
<label>Forecast +18 Hours</label>
<description>This is the weather forecast in 18 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus19" typeId="site-hr-forecast-grp">
<label>Forecast +19 Hours</label>
<description>This is the weather forecast in 19 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus20" typeId="site-hr-forecast-grp">
<label>Forecast +20 Hours</label>
<description>This is the weather forecast in 20 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus21" typeId="site-hr-forecast-grp">
<label>Forecast +21 Hours</label>
<description>This is the weather forecast in 21 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus22" typeId="site-hr-forecast-grp">
<label>Forecast +22 Hours</label>
<description>This is the weather forecast in 22 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus23" typeId="site-hr-forecast-grp">
<label>Forecast +23 Hours</label>
<description>This is the weather forecast in 23 hours.</description>
</channel-group>
<channel-group id="current-forecast-plus24" typeId="site-hr-forecast-grp">
<label>Forecast +24 Hours</label>
<description>This is the weather forecast in 24 hours.</description>
</channel-group>
<channel-group id="daily-forecast" typeId="site-daily-forecast-grp">
<label>Forecast Current Day</label>
<description>This is the weather forecast for the current day.</description>
</channel-group>
<channel-group id="daily-forecast-plus01" typeId="site-daily-forecast-grp">
<label>Forecast +1 Day</label>
<description>This is the weather forecast in 1 day.</description>
</channel-group>
<channel-group id="daily-forecast-plus02" typeId="site-daily-forecast-grp">
<label>Forecast +2 Days</label>
<description>This is the weather forecast in 2 days.</description>
</channel-group>
<channel-group id="daily-forecast-plus03" typeId="site-daily-forecast-grp">
<label>Forecast +3 Days</label>
<description>This is the weather forecast in 3 days.</description>
</channel-group>
<channel-group id="daily-forecast-plus04" typeId="site-daily-forecast-grp">
<label>Forecast +4 Days</label>
<description>This is the weather forecast in 4 days.</description>
</channel-group>
<channel-group id="daily-forecast-plus05" typeId="site-daily-forecast-grp">
<label>Forecast +5 Days</label>
<description>This is the weather forecast in 5 days.</description>
</channel-group>
<channel-group id="daily-forecast-plus06" typeId="site-daily-forecast-grp">
<label>Forecast +6 Days</label>
<description>This is the weather forecast in 6 days.</description>
</channel-group>
</channel-groups>
<config-description>
<parameter name="location" type="text">
<context>location</context>
<label>Weather Location</label>
<description>Location of weather forecast in geographical coordinates (latitude/longitude).</description>
</parameter>
<parameter name="hourlyForecastPollRate" type="integer" min="1" max="24">
<label>Hourly Poll Interval</label>
<description>The default number of hours to wait between retrieving hourly forecast data</description>
<default>1</default>
<advanced>true</advanced>
</parameter>
<parameter name="dailyForecastPollRate" type="integer" min="1" max="24">
<label>Daily Poll Interval</label>
<description>The default number of hours to wait between retrieving daily forecast data</description>
<default>3</default>
<advanced>true</advanced>
</parameter>
</config-description>
</thing-type>
</thing:thing-descriptions>

View File

@ -0,0 +1,93 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.api;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jetty.client.HttpClient;
import org.eclipse.jetty.client.HttpResponse;
import org.eclipse.jetty.client.api.Request;
import org.eclipse.jetty.client.api.Response;
import org.eclipse.jetty.client.api.Result;
import org.junit.jupiter.api.Test;
import java.util.List;
import static org.junit.jupiter.api.Assertions.assertDoesNotThrow;
import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertThrowsExactly;
import static org.junit.jupiter.api.Assertions.assertTrue;
/**
* The {@link SiteApiAuthenticationTest} class implements unit test case for {@link SiteApiAuthenticationTest}
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public class SiteApiAuthenticationTest {
private static Request dummyRequest = new HttpClient().newRequest("http://127.0.0.1:9999");;
@Test
public void testInitialNonAuthenticatedState() {
SiteApiAuthentication saa = new SiteApiAuthentication();
assertFalse(saa.getIsAuthenticated());
}
@Test
public void testAuthenticatedProcessing() {
SiteApiAuthentication saa = new SiteApiAuthentication();
Result result = new Result(dummyRequest,getResultWithStatus(200));
saa.processResult(result);
assertTrue(saa.getIsAuthenticated());
}
@Test
public void testUnauthenticatedProcessing() {
SiteApiAuthentication saa = new SiteApiAuthentication();
Result result = new Result(dummyRequest,getResultWithStatus(403));
saa.processResult(result);
assertFalse(saa.getIsAuthenticated());
}
@Test
public void testIsAuthenticatedUpdates() {
SiteApiAuthentication saa = new SiteApiAuthentication();
Result goodResult = new Result(dummyRequest,getResultWithStatus(200));
Result badresult = new Result(dummyRequest,getResultWithStatus(403));
saa.processResult(goodResult);
assertTrue(saa.getIsAuthenticated());
saa.processResult(badresult);
assertFalse(saa.getIsAuthenticated());
saa.processResult(goodResult);
assertTrue(saa.getIsAuthenticated());
}
@Test
public void testBadJwtDetected() {
SiteApiAuthentication saa = new SiteApiAuthentication();
assertThrowsExactly(AuthTokenException.class, () -> { saa.setApiKey("");});
assertThrowsExactly(AuthTokenException.class, () -> { saa.setApiKey("someInvalidToken.part");});
assertThrowsExactly(AuthTokenException.class, () -> { saa.setApiKey("\"someInvalidToken.part.new");});
}
@Test
public void testGoodJwtDetected() {
SiteApiAuthentication saa = new SiteApiAuthentication();
assertDoesNotThrow(() -> { saa.setApiKey("eyJhbGciOiJIUzI1NiJ9.eyJSb2xlIjoiQWRtaW4iLCJJc3N1ZXIiOiJJc3N1ZXIiLCJVc2VybmFtZSI6IkphdmFJblVzZSIsImV4cCI6MTczMDg1Mzg2NiwiaWF0IjoxNzMwODUzODY2fQ.Wmfe4npC037y0uoW4dnhizSXOPqFSn3OI3XbeklVQkA");});
}
private Response getResultWithStatus(final int status) {
return new HttpResponse(dummyRequest, List.of()).status(status);
}
}

View File

@ -0,0 +1,811 @@
{
"type": "FeatureCollection",
"features": [
{
"type": "Feature",
"geometry": {
"type": "Point",
"coordinates": [
-0.32430000000000003,
51.0624,
50.0
]
},
"properties": {
"location": {
"name": "Horsham"
},
"requestPointDistance": 0.1508,
"modelRunDate": "2022-09-19T21:00Z",
"timeSeries": [
{
"time": "2022-09-18T00:00Z",
"midnight10MWindSpeed": 1.39,
"midnight10MWindDirection": 31,
"midnight10MWindGust": 5.66,
"midnightVisibility": 8776,
"midnightRelativeHumidity": 91.42,
"midnightMslp": 102310,
"nightSignificantWeatherCode": 2,
"nightMinScreenTemperature": 8.05,
"nightUpperBoundMinTemp": 12.79,
"nightLowerBoundMinTemp": 5.51,
"nightMinFeelsLikeTemp": 7.35,
"nightUpperBoundMinFeelsLikeTemp": 12.83,
"nightLowerBoundMinFeelsLikeTemp": 7.33,
"nightProbabilityOfPrecipitation": 5,
"nightProbabilityOfSnow": 0,
"nightProbabilityOfHeavySnow": 1,
"nightProbabilityOfRain": 5,
"nightProbabilityOfHeavyRain": 3,
"nightProbabilityOfHail": 4,
"nightProbabilityOfSferics": 6
},
{
"time": "2022-09-19T00:00Z",
"midday10MWindSpeed": 1.03,
"midnight10MWindSpeed": 1.81,
"midday10MWindDirection": 299,
"midnight10MWindDirection": 323,
"midday10MWindGust": 2.06,
"midnight10MWindGust": 4.05,
"middayVisibility": 28034,
"midnightVisibility": 20277,
"middayRelativeHumidity": 63.54,
"midnightRelativeHumidity": 80.83,
"middayMslp": 102519,
"midnightMslp": 102541,
"maxUvIndex": 3,
"daySignificantWeatherCode": 7,
"nightSignificantWeatherCode": 7,
"dayMaxScreenTemperature": 17.64,
"nightMinScreenTemperature": 10.93,
"dayUpperBoundMaxTemp": 18.51,
"nightUpperBoundMinTemp": 13.74,
"dayLowerBoundMaxTemp": 14.22,
"nightLowerBoundMinTemp": 6.96,
"dayMaxFeelsLikeTemp": 17.06,
"nightMinFeelsLikeTemp": 11.12,
"dayUpperBoundMaxFeelsLikeTemp": 17.87,
"nightUpperBoundMinFeelsLikeTemp": 13.75,
"dayLowerBoundMaxFeelsLikeTemp": 14.57,
"nightLowerBoundMinFeelsLikeTemp": 9.81,
"dayProbabilityOfPrecipitation": 9,
"nightProbabilityOfPrecipitation": 6,
"dayProbabilityOfSnow": 0,
"nightProbabilityOfSnow": 0,
"dayProbabilityOfHeavySnow": 0,
"nightProbabilityOfHeavySnow": 0,
"dayProbabilityOfRain": 9,
"nightProbabilityOfRain": 6,
"dayProbabilityOfHeavyRain": 0,
"nightProbabilityOfHeavyRain": 1,
"dayProbabilityOfHail": 0,
"nightProbabilityOfHail": 0,
"dayProbabilityOfSferics": 0,
"nightProbabilityOfSferics": 0
},
{
"time": "2022-09-20T00:00Z",
"midday10MWindSpeed": 1.55,
"midnight10MWindSpeed": 1.02,
"midday10MWindDirection": 10,
"midnight10MWindDirection": 304,
"midday10MWindGust": 3.11,
"midnight10MWindGust": 2.33,
"middayVisibility": 28838,
"midnightVisibility": 16314,
"middayRelativeHumidity": 51.07,
"midnightRelativeHumidity": 83.33,
"middayMslp": 102680,
"midnightMslp": 102713,
"maxUvIndex": 3,
"daySignificantWeatherCode": 7,
"nightSignificantWeatherCode": 2,
"dayMaxScreenTemperature": 18.09,
"nightMinScreenTemperature": 9.92,
"dayUpperBoundMaxTemp": 19.61,
"nightUpperBoundMinTemp": 14.7,
"dayLowerBoundMaxTemp": 16.79,
"nightLowerBoundMinTemp": 5.69,
"dayMaxFeelsLikeTemp": 16.98,
"nightMinFeelsLikeTemp": 9.91,
"dayUpperBoundMaxFeelsLikeTemp": 18.95,
"nightUpperBoundMinFeelsLikeTemp": 14.99,
"dayLowerBoundMaxFeelsLikeTemp": 15.98,
"nightLowerBoundMinFeelsLikeTemp": 8.69,
"dayProbabilityOfPrecipitation": 5,
"nightProbabilityOfPrecipitation": 4,
"dayProbabilityOfSnow": 0,
"nightProbabilityOfSnow": 0,
"dayProbabilityOfHeavySnow": 0,
"nightProbabilityOfHeavySnow": 0,
"dayProbabilityOfRain": 5,
"nightProbabilityOfRain": 4,
"dayProbabilityOfHeavyRain": 0,
"nightProbabilityOfHeavyRain": 0,
"dayProbabilityOfHail": 0,
"nightProbabilityOfHail": 0,
"dayProbabilityOfSferics": 0,
"nightProbabilityOfSferics": 0
},
{
"time": "2022-09-21T00:00Z",
"midday10MWindSpeed": 1.05,
"midnight10MWindSpeed": 1.6,
"midday10MWindDirection": 201,
"midnight10MWindDirection": 133,
"midday10MWindGust": 2.85,
"midnight10MWindGust": 2.7,
"middayVisibility": 28467,
"midnightVisibility": 20397,
"middayRelativeHumidity": 51.33,
"midnightRelativeHumidity": 90.57,
"middayMslp": 102640,
"midnightMslp": 102440,
"maxUvIndex": 4,
"daySignificantWeatherCode": 1,
"nightSignificantWeatherCode": 2,
"dayMaxScreenTemperature": 19.75,
"nightMinScreenTemperature": 9.04,
"dayUpperBoundMaxTemp": 20.85,
"nightUpperBoundMinTemp": 12.07,
"dayLowerBoundMaxTemp": 17.71,
"nightLowerBoundMinTemp": 5.93,
"dayMaxFeelsLikeTemp": 18.82,
"nightMinFeelsLikeTemp": 8.75,
"dayUpperBoundMaxFeelsLikeTemp": 20.07,
"nightUpperBoundMinFeelsLikeTemp": 11.3,
"dayLowerBoundMaxFeelsLikeTemp": 17.23,
"nightLowerBoundMinFeelsLikeTemp": 5.31,
"dayProbabilityOfPrecipitation": 1,
"nightProbabilityOfPrecipitation": 2,
"dayProbabilityOfSnow": 0,
"nightProbabilityOfSnow": 0,
"dayProbabilityOfHeavySnow": 0,
"nightProbabilityOfHeavySnow": 0,
"dayProbabilityOfRain": 1,
"nightProbabilityOfRain": 2,
"dayProbabilityOfHeavyRain": 0,
"nightProbabilityOfHeavyRain": 0,
"dayProbabilityOfHail": 0,
"nightProbabilityOfHail": 0,
"dayProbabilityOfSferics": 0,
"nightProbabilityOfSferics": 0
},
{
"time": "2022-09-22T00:00Z",
"midday10MWindSpeed": 3.87,
"midnight10MWindSpeed": 2.21,
"midday10MWindDirection": 192,
"midnight10MWindDirection": 167,
"midday10MWindGust": 8.14,
"midnight10MWindGust": 3.35,
"middayVisibility": 32675,
"midnightVisibility": 25210,
"middayRelativeHumidity": 61.39,
"midnightRelativeHumidity": 88.29,
"middayMslp": 102164,
"midnightMslp": 101816,
"maxUvIndex": 3,
"daySignificantWeatherCode": 7,
"nightSignificantWeatherCode": 7,
"dayMaxScreenTemperature": 18.71,
"nightMinScreenTemperature": 10.89,
"dayUpperBoundMaxTemp": 21.93,
"nightUpperBoundMinTemp": 14.6,
"dayLowerBoundMaxTemp": 17.77,
"nightLowerBoundMinTemp": 7.75,
"dayMaxFeelsLikeTemp": 16.78,
"nightMinFeelsLikeTemp": 10.78,
"dayUpperBoundMaxFeelsLikeTemp": 19.63,
"nightUpperBoundMinFeelsLikeTemp": 13.97,
"dayLowerBoundMaxFeelsLikeTemp": 16.54,
"nightLowerBoundMinFeelsLikeTemp": 7.46,
"dayProbabilityOfPrecipitation": 5,
"nightProbabilityOfPrecipitation": 9,
"dayProbabilityOfSnow": 0,
"nightProbabilityOfSnow": 0,
"dayProbabilityOfHeavySnow": 0,
"nightProbabilityOfHeavySnow": 0,
"dayProbabilityOfRain": 5,
"nightProbabilityOfRain": 9,
"dayProbabilityOfHeavyRain": 0,
"nightProbabilityOfHeavyRain": 4,
"dayProbabilityOfHail": 0,
"nightProbabilityOfHail": 1,
"dayProbabilityOfSferics": 0,
"nightProbabilityOfSferics": 1
},
{
"time": "2022-09-23T00:00Z",
"midday10MWindSpeed": 2.42,
"midnight10MWindSpeed": 1.73,
"midday10MWindDirection": 209,
"midnight10MWindDirection": 306,
"midday10MWindGust": 5.59,
"midnight10MWindGust": 3.31,
"middayVisibility": 30194,
"midnightVisibility": 9598,
"middayRelativeHumidity": 63.53,
"midnightRelativeHumidity": 92.99,
"middayMslp": 101628,
"midnightMslp": 101583,
"maxUvIndex": 3,
"daySignificantWeatherCode": 7,
"nightSignificantWeatherCode": 12,
"dayMaxScreenTemperature": 18.76,
"nightMinScreenTemperature": 11.6,
"dayUpperBoundMaxTemp": 21.29,
"nightUpperBoundMinTemp": 13.67,
"dayLowerBoundMaxTemp": 14.51,
"nightLowerBoundMinTemp": 6.44,
"dayMaxFeelsLikeTemp": 17.49,
"nightMinFeelsLikeTemp": 11.38,
"dayUpperBoundMaxFeelsLikeTemp": 20.03,
"nightUpperBoundMinFeelsLikeTemp": 13.12,
"dayLowerBoundMaxFeelsLikeTemp": 14.39,
"nightLowerBoundMinFeelsLikeTemp": 5.45,
"dayProbabilityOfPrecipitation": 43,
"nightProbabilityOfPrecipitation": 45,
"dayProbabilityOfSnow": 0,
"nightProbabilityOfSnow": 0,
"dayProbabilityOfHeavySnow": 0,
"nightProbabilityOfHeavySnow": 0,
"dayProbabilityOfRain": 43,
"nightProbabilityOfRain": 45,
"dayProbabilityOfHeavyRain": 22,
"nightProbabilityOfHeavyRain": 26,
"dayProbabilityOfHail": 1,
"nightProbabilityOfHail": 1,
"dayProbabilityOfSferics": 2,
"nightProbabilityOfSferics": 2
},
{
"time": "2022-09-24T00:00Z",
"midday10MWindSpeed": 3.95,
"midnight10MWindSpeed": 4.28,
"midday10MWindDirection": 43,
"midnight10MWindDirection": 6,
"midday10MWindGust": 8.46,
"midnight10MWindGust": 8.19,
"middayVisibility": 17042,
"midnightVisibility": 26745,
"middayRelativeHumidity": 74.83,
"midnightRelativeHumidity": 81.26,
"middayMslp": 101763,
"midnightMslp": 101959,
"maxUvIndex": 3,
"daySignificantWeatherCode": 7,
"nightSignificantWeatherCode": 0,
"dayMaxScreenTemperature": 17.4,
"nightMinScreenTemperature": 11.09,
"dayUpperBoundMaxTemp": 22.92,
"nightUpperBoundMinTemp": 13.81,
"dayLowerBoundMaxTemp": 13.72,
"nightLowerBoundMinTemp": 4.24,
"dayMaxFeelsLikeTemp": 15.15,
"nightMinFeelsLikeTemp": 9.74,
"dayUpperBoundMaxFeelsLikeTemp": 21.94,
"nightUpperBoundMinFeelsLikeTemp": 13.29,
"dayLowerBoundMaxFeelsLikeTemp": 11.64,
"nightLowerBoundMinFeelsLikeTemp": 3.44,
"dayProbabilityOfPrecipitation": 32,
"nightProbabilityOfPrecipitation": 10,
"dayProbabilityOfSnow": 0,
"nightProbabilityOfSnow": 0,
"dayProbabilityOfHeavySnow": 0,
"nightProbabilityOfHeavySnow": 0,
"dayProbabilityOfRain": 32,
"nightProbabilityOfRain": 10,
"dayProbabilityOfHeavyRain": 17,
"nightProbabilityOfHeavyRain": 6,
"dayProbabilityOfHail": 1,
"nightProbabilityOfHail": 0,
"dayProbabilityOfSferics": 4,
"nightProbabilityOfSferics": 1
},
{
"time": "2022-09-25T00:00Z",
"midday10MWindSpeed": 4.36,
"midnight10MWindSpeed": 2.77,
"midday10MWindDirection": 32,
"midnight10MWindDirection": 269,
"midday10MWindGust": 9.45,
"midnight10MWindGust": 4.63,
"middayVisibility": 31004,
"midnightVisibility": 24820,
"middayRelativeHumidity": 60.24,
"midnightRelativeHumidity": 88.87,
"middayMslp": 102022,
"midnightMslp": 101654,
"maxUvIndex": 3,
"daySignificantWeatherCode": 3,
"nightSignificantWeatherCode": 2,
"dayMaxScreenTemperature": 17.65,
"nightMinScreenTemperature": 8.73,
"dayUpperBoundMaxTemp": 22.27,
"nightUpperBoundMinTemp": 13.54,
"dayLowerBoundMaxTemp": 11.66,
"nightLowerBoundMinTemp": 4.02,
"dayMaxFeelsLikeTemp": 15.0,
"nightMinFeelsLikeTemp": 7.98,
"dayUpperBoundMaxFeelsLikeTemp": 21.24,
"nightUpperBoundMinFeelsLikeTemp": 12.32,
"dayLowerBoundMaxFeelsLikeTemp": 11.09,
"nightLowerBoundMinFeelsLikeTemp": 3.6,
"dayProbabilityOfPrecipitation": 9,
"nightProbabilityOfPrecipitation": 6,
"dayProbabilityOfSnow": 0,
"nightProbabilityOfSnow": 0,
"dayProbabilityOfHeavySnow": 0,
"nightProbabilityOfHeavySnow": 0,
"dayProbabilityOfRain": 9,
"nightProbabilityOfRain": 6,
"dayProbabilityOfHeavyRain": 3,
"nightProbabilityOfHeavyRain": 4,
"dayProbabilityOfHail": 0,
"nightProbabilityOfHail": 1,
"dayProbabilityOfSferics": 1,
"nightProbabilityOfSferics": 1
}
]
}
}
],
"parameters": [
{
"daySignificantWeatherCode": {
"type": "Parameter",
"description": "DaySignificantWeatherCode",
"unit": {
"label": "dimensionless",
"symbol": {
"value": "https://metoffice.apiconnect.ibmcloud.com/metoffice/production/",
"type": "1"
}
}
},
"midnightRelativeHumidity": {
"type": "Parameter",
"description": "RelativeHumidityatLocalMidnight",
"unit": {
"label": "percentage",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "%"
}
}
},
"nightProbabilityOfHeavyRain": {
"type": "Parameter",
"description": "ProbabilityofHeavyRainDuringTheNight",
"unit": {
"label": "percentage",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "%"
}
}
},
"midnight10MWindSpeed": {
"type": "Parameter",
"description": "10mWindSpeedatLocalMidnight",
"unit": {
"label": "metrespersecond",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "m/s"
}
}
},
"nightUpperBoundMinFeelsLikeTemp": {
"type": "Parameter",
"description": "UpperBoundonNightMinimumFeelsLikeAirTemperature",
"unit": {
"label": "degreesCelsius",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "Cel"
}
}
},
"nightUpperBoundMinTemp": {
"type": "Parameter",
"description": "UpperBoundonNightMinimumScreenAirTemperature",
"unit": {
"label": "degreesCelsius",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "Cel"
}
}
},
"midnightVisibility": {
"type": "Parameter",
"description": "VisibilityatLocalMidnight",
"unit": {
"label": "metres",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "m"
}
}
},
"dayUpperBoundMaxFeelsLikeTemp": {
"type": "Parameter",
"description": "UpperBoundonDayMaximumFeelsLikeAirTemperature",
"unit": {
"label": "degreesCelsius",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "Cel"
}
}
},
"nightProbabilityOfRain": {
"type": "Parameter",
"description": "ProbabilityofRainDuringTheNight",
"unit": {
"label": "percentage",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "%"
}
}
},
"midday10MWindDirection": {
"type": "Parameter",
"description": "10mWindDirectionatLocalMidday",
"unit": {
"label": "degrees",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "deg"
}
}
},
"nightLowerBoundMinFeelsLikeTemp": {
"type": "Parameter",
"description": "LowerBoundonNightMinimumFeelsLikeAirTemperature",
"unit": {
"label": "degreesCelsius",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "Cel"
}
}
},
"nightProbabilityOfHail": {
"type": "Parameter",
"description": "ProbabilityofHailDuringTheNight",
"unit": {
"label": "percentage",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "%"
}
}
},
"middayMslp": {
"type": "Parameter",
"description": "MeanSeaLevelPressureatLocalMidday",
"unit": {
"label": "pascals",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "Pa"
}
}
},
"dayProbabilityOfHeavySnow": {
"type": "Parameter",
"description": "ProbabilityofHeavySnowDuringTheDay",
"unit": {
"label": "percentage",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "%"
}
}
},
"nightProbabilityOfPrecipitation": {
"type": "Parameter",
"description": "ProbabilityofPrecipitationDuringTheNight",
"unit": {
"label": "percentage",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "%"
}
}
},
"dayProbabilityOfHail": {
"type": "Parameter",
"description": "ProbabilityofHailDuringTheDay",
"unit": {
"label": "percentage",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "%"
}
}
},
"dayProbabilityOfRain": {
"type": "Parameter",
"description": "ProbabilityofRainDuringTheDay",
"unit": {
"label": "percentage",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "%"
}
}
},
"midday10MWindSpeed": {
"type": "Parameter",
"description": "10mWindSpeedatLocalMidday",
"unit": {
"label": "metrespersecond",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "m/s"
}
}
},
"midday10MWindGust": {
"type": "Parameter",
"description": "10mWindGustSpeedatLocalMidday",
"unit": {
"label": "metrespersecond",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "m/s"
}
}
},
"middayVisibility": {
"type": "Parameter",
"description": "VisibilityatLocalMidday",
"unit": {
"label": "metres",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "m"
}
}
},
"midnight10MWindGust": {
"type": "Parameter",
"description": "10mWindGustSpeedatLocalMidnight",
"unit": {
"label": "metrespersecond",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "m/s"
}
}
},
"midnightMslp": {
"type": "Parameter",
"description": "MeanSeaLevelPressureatLocalMidnight",
"unit": {
"label": "pascals",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "Pa"
}
}
},
"dayProbabilityOfSferics": {
"type": "Parameter",
"description": "ProbabilityofSfericsDuringTheDay",
"unit": {
"label": "percentage",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "%"
}
}
},
"nightSignificantWeatherCode": {
"type": "Parameter",
"description": "NightSignificantWeatherCode",
"unit": {
"label": "dimensionless",
"symbol": {
"value": "https://metoffice.apiconnect.ibmcloud.com/metoffice/production/",
"type": "1"
}
}
},
"dayProbabilityOfPrecipitation": {
"type": "Parameter",
"description": "ProbabilityofPrecipitationDuringTheDay",
"unit": {
"label": "percentage",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "%"
}
}
},
"dayProbabilityOfHeavyRain": {
"type": "Parameter",
"description": "ProbabilityofHeavyRainDuringTheDay",
"unit": {
"label": "percentage",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "%"
}
}
},
"dayMaxScreenTemperature": {
"type": "Parameter",
"description": "DayMaximumScreenAirTemperature",
"unit": {
"label": "degreesCelsius",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "Cel"
}
}
},
"nightMinScreenTemperature": {
"type": "Parameter",
"description": "NightMinimumScreenAirTemperature",
"unit": {
"label": "degreesCelsius",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "Cel"
}
}
},
"midnight10MWindDirection": {
"type": "Parameter",
"description": "10mWindDirectionatLocalMidnight",
"unit": {
"label": "degrees",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "deg"
}
}
},
"maxUvIndex": {
"type": "Parameter",
"description": "DayMaximumUVIndex",
"unit": {
"label": "dimensionless",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "1"
}
}
},
"dayProbabilityOfSnow": {
"type": "Parameter",
"description": "ProbabilityofSnowDuringTheDay",
"unit": {
"label": "percentage",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "%"
}
}
},
"nightProbabilityOfSnow": {
"type": "Parameter",
"description": "ProbabilityofSnowDuringTheNight",
"unit": {
"label": "percentage",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "%"
}
}
},
"dayLowerBoundMaxTemp": {
"type": "Parameter",
"description": "LowerBoundonDayMaximumScreenAirTemperature",
"unit": {
"label": "degreesCelsius",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "Cel"
}
}
},
"nightProbabilityOfHeavySnow": {
"type": "Parameter",
"description": "ProbabilityofHeavySnowDuringTheNight",
"unit": {
"label": "percentage",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "%"
}
}
},
"dayLowerBoundMaxFeelsLikeTemp": {
"type": "Parameter",
"description": "LowerBoundonDayMaximumFeelsLikeAirTemperature",
"unit": {
"label": "degreesCelsius",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "Cel"
}
}
},
"dayUpperBoundMaxTemp": {
"type": "Parameter",
"description": "UpperBoundonDayMaximumScreenAirTemperature",
"unit": {
"label": "degreesCelsius",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "Cel"
}
}
},
"dayMaxFeelsLikeTemp": {
"type": "Parameter",
"description": "DayMaximumFeelsLikeAirTemperature",
"unit": {
"label": "degreesCelsius",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "Cel"
}
}
},
"middayRelativeHumidity": {
"type": "Parameter",
"description": "RelativeHumidityatLocalMidday",
"unit": {
"label": "percentage",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "%"
}
}
},
"nightLowerBoundMinTemp": {
"type": "Parameter",
"description": "LowerBoundonNightMinimumScreenAirTemperature",
"unit": {
"label": "degreesCelsius",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "Cel"
}
}
},
"nightMinFeelsLikeTemp": {
"type": "Parameter",
"description": "NightMinimumFeelsLikeAirTemperature",
"unit": {
"label": "degreesCelsius",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "Cel"
}
}
},
"nightProbabilityOfSferics": {
"type": "Parameter",
"description": "ProbabilityofSfericsDuringTheNight",
"unit": {
"label": "percentage",
"symbol": {
"value": "http://www.opengis.net/def/uom/UCUM/",
"type": "%"
}
}
}
}
]
}

View File

@ -0,0 +1,238 @@
/**
* Copyright (c) 2010-2024 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.metofficedatahub.internal.dto.responses;
import static org.junit.jupiter.api.Assertions.*;
import static org.junit.jupiter.api.Assertions.assertNotNull;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
import org.junit.jupiter.api.Test;
import org.openhab.binding.metofficedatahub.internal.MetOfficeDataHubBindingConstants;
/**
* The {@link SiteApiFeatureCollectionTest} class implements unit test case for {@link SiteApiFeatureCollection}
*
* @author David Goodyear - Initial contribution
*/
@NonNullByDefault
public class SiteApiFeatureCollectionTest {
private @Nullable String siteApiDailyResponse = null;
private @Nullable String siteApiHourlyResponse = null;
public @Nullable String getSiteDailyApiResponse() {
try {
if (siteApiDailyResponse == null) {
java.net.URL url = SiteApiFeatureCollectionTest.class.getResource("2022-09-siteDailyResponse.json");
if (url == null) {
return null;
}
java.nio.file.Path resPath = java.nio.file.Paths.get(url.toURI());
siteApiDailyResponse = new String(java.nio.file.Files.readAllBytes(resPath), "UTF8");
}
} catch (Exception e) {
return null;
}
return siteApiDailyResponse;
}
public @Nullable String getSiteHourlyApiResponse() {
try {
if (siteApiHourlyResponse == null) {
java.net.URL url = SiteApiFeatureCollectionTest.class.getResource("2022-09-siteHourlyResponse.json");
if (url == null) {
return null;
}
java.nio.file.Path resPath = java.nio.file.Paths.get(url.toURI());
siteApiHourlyResponse = new String(java.nio.file.Files.readAllBytes(resPath), "UTF8");
}
} catch (Exception e) {
return null;
}
return siteApiHourlyResponse;
}
@Test
public void testSiteApiFeatureCollectionHourly() {
SiteApiFeatureCollection response = MetOfficeDataHubBindingConstants.GSON.fromJson(getSiteHourlyApiResponse(),
SiteApiFeatureCollection.class);
if (response != null) {
assertEquals(SiteApiFeatureCollection.TYPE_SITE_API_FEATURE_COLLECTION, response.getType());
} else {
fail("GSON returned null");
}
}
@Test
public void testSiteApiFeatureCollectionDaily() {
SiteApiFeatureCollection response = MetOfficeDataHubBindingConstants.GSON.fromJson(getSiteDailyApiResponse(),
SiteApiFeatureCollection.class);
if (response != null) {
assertEquals(SiteApiFeatureCollection.TYPE_SITE_API_FEATURE_COLLECTION, response.getType());
} else {
fail("GSON returned null");
}
}
@Test
public void testSiteApiFeatureHourly() {
SiteApiFeatureCollection response = MetOfficeDataHubBindingConstants.GSON.fromJson(getSiteHourlyApiResponse(),
SiteApiFeatureCollection.class);
assertNotNull(response);
assertNotNull(response.getFeature());
assertEquals(1, response.getFeature().length);
assertEquals(SiteApiFeature.TYPE_SITE_API_FEATURE, response.getFeature()[0].getType());
}
@Test
public void testSiteApiFeatureDaily() {
SiteApiFeatureCollection response = MetOfficeDataHubBindingConstants.GSON.fromJson(getSiteDailyApiResponse(),
SiteApiFeatureCollection.class);
assertNotNull(response);
assertNotNull(response.getFeature());
assertEquals(1,response.getFeature().length);
assertEquals(SiteApiFeature.TYPE_SITE_API_FEATURE, response.getFeature()[0].getType());
}
@Test
public void testSiteApiFeatureGeometryHourly() {
SiteApiFeatureCollection response = MetOfficeDataHubBindingConstants.GSON.fromJson(getSiteHourlyApiResponse(),
SiteApiFeatureCollection.class);
assertNotNull(response);
assertNotNull(response.getFeature());
assertEquals(1, response.getFeature().length);
assertEquals(SiteApiFeaturePoint.TYPE_SITE_API_FEATURE, response.getFeature()[0].getGeometry().getType());
assertEquals(-0.32430000000000003,response.getFeature()[0].getGeometry().getLongitude());
assertEquals(51.0624,response.getFeature()[0].getGeometry().getLatitude());
assertEquals(50.0,response.getFeature()[0].getGeometry().getElevation());
}
@Test
public void testSiteApiFeatureGeometryDaily() {
SiteApiFeatureCollection response = MetOfficeDataHubBindingConstants.GSON.fromJson(getSiteDailyApiResponse(),
SiteApiFeatureCollection.class);
assertNotNull(response);
assertNotNull(response.getFeature());
assertEquals(1, response.getFeature().length);
assertEquals(SiteApiFeaturePoint.TYPE_SITE_API_FEATURE, response.getFeature()[0].getGeometry().getType());
assertEquals(-0.32430000000000003,response.getFeature()[0].getGeometry().getLongitude());
assertEquals(51.0624,response.getFeature()[0].getGeometry().getLatitude());
assertEquals(50.0,response.getFeature()[0].getGeometry().getElevation());
}
@Test
public void testSiteApiFeaturePropertiesHourly() {
SiteApiFeatureCollection response = MetOfficeDataHubBindingConstants.GSON.fromJson(getSiteHourlyApiResponse(),
SiteApiFeatureCollection.class);
assertNotNull(response);
assertNotNull(response.getFeature());
assertEquals(1, response.getFeature().length);
assertEquals("Horsham",response.getFeature()[0].getProperties().getLocation().getName());
assertEquals(0.1508,response.getFeature()[0].getProperties().getRequestPointDistance());
assertEquals("2022-09-17T20:00Z",response.getFeature()[0].getProperties().getModelRunDate());
}
@Test
public void testSiteApiFeaturePropertiesDaily() {
SiteApiFeatureCollection response = MetOfficeDataHubBindingConstants.GSON.fromJson(getSiteDailyApiResponse(),
SiteApiFeatureCollection.class);
assertNotNull(response);
assertNotNull(response.getFeature());
assertEquals(1, response.getFeature().length);
assertEquals("Horsham",response.getFeature()[0].getProperties().getLocation().getName());
assertEquals(0.1508,response.getFeature()[0].getProperties().getRequestPointDistance());
assertEquals("2022-09-19T21:00Z",response.getFeature()[0].getProperties().getModelRunDate());
}
@Test
public void testSiteApiFeaturePropertiesTimeSeries0Hourly() {
SiteApiFeatureCollection response = MetOfficeDataHubBindingConstants.GSON.fromJson(getSiteHourlyApiResponse(),
SiteApiFeatureCollection.class);
assertNotNull(response);
assertNotNull(response.getFeature());
assertEquals(1, response.getFeature().length);
assertEquals("2022-09-17T20:00Z",response.getFeature()[0].getProperties().getTimeSeries()[0].getTime());
assertEquals( 8.55,response.getFeature()[0].getProperties().getTimeSeries()[0].getScreenTemperature());
assertEquals( 10.36,response.getFeature()[0].getProperties().getTimeSeries()[0].getMaxScreenTemperature());
assertEquals( 8.54,response.getFeature()[0].getProperties().getTimeSeries()[0].getMinScreenTemperature());
assertEquals( 4.67,response.getFeature()[0].getProperties().getTimeSeries()[0].getScreenDewPointTemperature());
assertEquals( 8.18,response.getFeature()[0].getProperties().getTimeSeries()[0].getFeelsLikeTemperature());
assertEquals( 0.46,response.getFeature()[0].getProperties().getTimeSeries()[0].getWindSpeed10m());
assertEquals( 297,response.getFeature()[0].getProperties().getTimeSeries()[0].getWindDirectionFrom10m());
assertEquals( 4.63,response.getFeature()[0].getProperties().getTimeSeries()[0].getWindGustSpeed10m());
assertEquals( 6.43,response.getFeature()[0].getProperties().getTimeSeries()[0].getMax10mWindGust());
assertEquals( 19040,response.getFeature()[0].getProperties().getTimeSeries()[0].getVisibility());
assertEquals( 76.51,response.getFeature()[0].getProperties().getTimeSeries()[0].getScreenRelativeHumidity());
assertEquals( 102230,response.getFeature()[0].getProperties().getTimeSeries()[0].getPressure());
assertEquals( 1,response.getFeature()[0].getProperties().getTimeSeries()[0].getUvIndex());
assertEquals( 2,response.getFeature()[0].getProperties().getTimeSeries()[0].getSignificantWeatherCode());
assertEquals( 3,response.getFeature()[0].getProperties().getTimeSeries()[0].getPrecipitationRate());
assertEquals( 4,response.getFeature()[0].getProperties().getTimeSeries()[0].getTotalPrecipAmount());
assertEquals( 5,response.getFeature()[0].getProperties().getTimeSeries()[0].getTotalSnowAmount());
assertEquals( 60,response.getFeature()[0].getProperties().getTimeSeries()[0].getProbOfPrecipitation());
}
@Test
public void testSiteApiFeaturePropertiesTimeSeries0Daily() {
SiteApiFeatureCollection response = MetOfficeDataHubBindingConstants.GSON.fromJson(getSiteDailyApiResponse(),
SiteApiFeatureCollection.class);
assertNotNull(response);
assertNotNull(response.getFeature());
assertEquals(1, response.getFeature().length);
assertEquals("2022-09-18T00:00Z",response.getFeature()[0].getProperties().getTimeSeries()[0].getTime());
assertEquals( 1.39,response.getFeature()[0].getProperties().getTimeSeries()[0].getMidnight10MWindSpeed());
assertEquals( 31,response.getFeature()[0].getProperties().getTimeSeries()[0].getMidnight10MWindDirection());
assertEquals( 5.66,response.getFeature()[0].getProperties().getTimeSeries()[0].getMidnight10MWindGust());
assertEquals( 8776,response.getFeature()[0].getProperties().getTimeSeries()[0].getMidnightVisibility());
assertEquals( 91.42,response.getFeature()[0].getProperties().getTimeSeries()[0].getMidnightRelativeHumidity());
assertEquals( 102310,response.getFeature()[0].getProperties().getTimeSeries()[0].getMidnightPressure());
assertEquals( 2,response.getFeature()[0].getProperties().getTimeSeries()[0].getNightSignificantWeatherCode());
assertEquals( 8.05,response.getFeature()[0].getProperties().getTimeSeries()[0].getNightMinScreenTemperature());
assertEquals( 12.79,response.getFeature()[0].getProperties().getTimeSeries()[0].getNightUpperBoundMinTemp());
assertEquals( 5.51,response.getFeature()[0].getProperties().getTimeSeries()[0].getNightLowerBoundMinTemp());
assertEquals( 7.35,response.getFeature()[0].getProperties().getTimeSeries()[0].getNightMinFeelsLikeTemp());
assertEquals( 12.83,response.getFeature()[0].getProperties().getTimeSeries()[0].getNightUpperBoundMinFeelsLikeTemp());
assertEquals( 7.33,response.getFeature()[0].getProperties().getTimeSeries()[0].getNightLowerBoundMinFeelsLikeTemp());
assertEquals( 5,response.getFeature()[0].getProperties().getTimeSeries()[0].getNightProbabilityOfPrecipitation());
assertEquals( 0,response.getFeature()[0].getProperties().getTimeSeries()[0].getNightProbabilityOfSnow());
assertEquals( 1,response.getFeature()[0].getProperties().getTimeSeries()[0].getNightProbabilityOfHeavySnow());
assertEquals( 5,response.getFeature()[0].getProperties().getTimeSeries()[0].getNightProbabilityOfRain());
assertEquals( 3,response.getFeature()[0].getProperties().getTimeSeries()[0].getNightProbabilityOfHeavyRain());
assertEquals( 4,response.getFeature()[0].getProperties().getTimeSeries()[0].getNightProbabilityOfHail());
assertEquals( 6,response.getFeature()[0].getProperties().getTimeSeries()[0].getNightProbabilityOfSferics());
}
@Test
public void testGetTimeSeriesForCurrentHour() {
SiteApiFeatureCollection response = MetOfficeDataHubBindingConstants.GSON.fromJson(getSiteHourlyApiResponse(),
SiteApiFeatureCollection.class);
assertNotNull(response);
assertNotNull(response.getFeature());
assertEquals(1, response.getFeature().length);
assertEquals( 0,response.getFeature()[0].getProperties().getHourlyTimeSeriesPositionForCurrentHour("2022-09-17T20:00Z"));
assertEquals( 1,response.getFeature()[0].getProperties().getHourlyTimeSeriesPositionForCurrentHour("2022-09-17T21:00Z"));
assertEquals( 2,response.getFeature()[0].getProperties().getHourlyTimeSeriesPositionForCurrentHour("2022-09-17T22:00Z"));
assertEquals( 3,response.getFeature()[0].getProperties().getHourlyTimeSeriesPositionForCurrentHour("2022-09-17T23:00Z"));
assertEquals( 24,response.getFeature()[0].getProperties().getHourlyTimeSeriesPositionForCurrentHour("2022-09-18T20:00Z"));
assertEquals( 48,response.getFeature()[0].getProperties().getHourlyTimeSeriesPositionForCurrentHour("2022-09-19T20:00Z"));
}
}

View File

@ -253,6 +253,7 @@
<module>org.openhab.binding.meteoblue</module>
<module>org.openhab.binding.meteofrance</module>
<module>org.openhab.binding.meteostick</module>
<module>org.openhab.binding.metofficedatahub</module>
<module>org.openhab.binding.mffan</module>
<module>org.openhab.binding.miele</module>
<module>org.openhab.binding.mielecloud</module>