6
6
from typing import List
7
7
8
8
import async_timeout
9
+ import pytz
9
10
from homeassistant .components .weather import Forecast
10
11
from homeassistant .const import ATTR_LATITUDE , ATTR_LONGITUDE
11
12
from homeassistant .helpers .aiohttp_client import async_get_clientsession
16
17
from .api import IrmKmiApiClient , IrmKmiApiError
17
18
from .const import IRM_KMI_TO_HA_CONDITION_MAP as CDT_MAP
18
19
from .const import OUT_OF_BENELUX
19
- from .data import IrmKmiForecast
20
+ from .data import (AnimationFrameData , CurrentWeatherData , IrmKmiForecast ,
21
+ ProcessedCoordinatorData , RadarAnimationData )
20
22
21
23
_LOGGER = logging .getLogger (__name__ )
22
24
@@ -37,7 +39,7 @@ def __init__(self, hass, zone):
37
39
self ._api_client = IrmKmiApiClient (session = async_get_clientsession (hass ))
38
40
self ._zone = zone
39
41
40
- async def _async_update_data (self ):
42
+ async def _async_update_data (self ) -> ProcessedCoordinatorData :
41
43
"""Fetch data from API endpoint.
42
44
43
45
This is the place to pre-process the data to lookup tables
@@ -62,67 +64,107 @@ async def _async_update_data(self):
62
64
if api_data .get ('cityName' , None ) in OUT_OF_BENELUX :
63
65
raise UpdateFailed (f"Zone '{ self ._zone } ' is out of Benelux and forecast is only available in the Benelux" )
64
66
65
- result = self .process_api_data (api_data )
67
+ return await self .process_api_data (api_data )
66
68
67
- # TODO make such that the most up to date image is specified to entity for static display
68
- return result | await self ._async_animation_data (api_data )
69
+ async def _async_animation_data (self , api_data : dict ) -> RadarAnimationData :
69
70
70
- async def _async_animation_data (self , api_data : dict ) -> dict :
71
-
72
- default = {'animation' : None }
73
71
animation_data = api_data .get ('animation' , {}).get ('sequence' )
74
- localisation_layer = api_data .get ('animation' , {}).get ('localisationLayer' )
75
- country = api_data .get ('country' , None )
72
+ localisation_layer_url = api_data .get ('animation' , {}).get ('localisationLayer' )
73
+ country = api_data .get ('country' , '' )
74
+
75
+ if animation_data is None or localisation_layer_url is None or not isinstance (animation_data , list ):
76
+ return RadarAnimationData ()
77
+
78
+ try :
79
+ images_from_api = await self .download_images_from_api (animation_data , country , localisation_layer_url )
80
+ except IrmKmiApiError :
81
+ _LOGGER .warning (f"Could not get images for weather radar" )
82
+ return RadarAnimationData ()
83
+
84
+ localisation = Image .open (BytesIO (images_from_api [0 ])).convert ('RGBA' )
85
+ images_from_api = images_from_api [1 :]
76
86
77
- if animation_data is None or localisation_layer is None or not isinstance (animation_data , list ):
78
- return default
87
+ radar_animation = await self .merge_frames_from_api (animation_data , country , images_from_api , localisation )
88
+ # TODO support translation here
89
+ radar_animation ['hint' ] = api_data .get ('animation' , {}).get ('sequenceHint' , {}).get ('en' )
90
+ return radar_animation
79
91
92
+ async def download_images_from_api (self , animation_data , country , localisation_layer_url ):
80
93
coroutines = list ()
81
- coroutines .append (self ._api_client .get_image (f"{ localisation_layer } &th={ 'd' if country == 'NL' else 'n' } " ))
94
+ coroutines .append (self ._api_client .get_image (f"{ localisation_layer_url } &th={ 'd' if country == 'NL' else 'n' } " ))
82
95
for frame in animation_data :
83
96
if frame .get ('uri' , None ) is not None :
84
97
coroutines .append (self ._api_client .get_image (frame .get ('uri' )))
98
+ async with async_timeout .timeout (20 ):
99
+ images_from_api = await asyncio .gather (* coroutines , return_exceptions = True )
85
100
86
- try :
87
- async with async_timeout .timeout (20 ):
88
- r = await asyncio .gather (* coroutines , return_exceptions = True )
89
- except IrmKmiApiError :
90
- _LOGGER .warning (f"Could not get images for weather radar" )
91
- return default
92
- _LOGGER .debug (f"Just downloaded { len (r )} images" )
101
+ _LOGGER .debug (f"Just downloaded { len (images_from_api )} images" )
102
+ return images_from_api
103
+
104
+ async def merge_frames_from_api (self , animation_data , country , images_from_api ,
105
+ localisation_layer ) -> RadarAnimationData :
93
106
94
107
if country == 'NL' :
95
108
background = Image .open ("custom_components/irm_kmi/resources/nl.png" ).convert ('RGBA' )
96
109
else :
97
110
background = Image .open ("custom_components/irm_kmi/resources/be_bw.png" ).convert ('RGBA' )
98
- localisation = Image .open (BytesIO (r [0 ])).convert ('RGBA' )
99
- merged_frames = list ()
100
- for frame in r [1 :]:
111
+
112
+ most_recent_frame = None
113
+ tz = pytz .timezone (self .hass .config .time_zone )
114
+ current_time = datetime .now (tz = tz )
115
+ sequence : List [AnimationFrameData ] = list ()
116
+ for (idx , sequence_element ) in enumerate (animation_data ):
117
+ frame = images_from_api [idx ]
101
118
layer = Image .open (BytesIO (frame )).convert ('RGBA' )
102
119
temp = Image .alpha_composite (background , layer )
103
- temp = Image .alpha_composite (temp , localisation )
120
+ temp = Image .alpha_composite (temp , localisation_layer )
104
121
105
122
draw = ImageDraw .Draw (temp )
106
123
font = ImageFont .truetype ("custom_components/irm_kmi/resources/roboto_medium.ttf" , 16 )
107
- # TODO write actual date time
124
+ time_image = (datetime .fromisoformat (sequence_element .get ('time' ))
125
+ .astimezone (tz = tz ))
126
+
127
+ time_str = time_image .isoformat (sep = ' ' , timespec = 'minutes' )
128
+
108
129
if country == 'NL' :
109
- draw .text ((4 , 4 ), "Sample Text" , (0 , 0 , 0 ), font = font )
130
+ draw .text ((4 , 4 ), time_str , (0 , 0 , 0 ), font = font )
110
131
else :
111
- draw .text ((4 , 4 ), "Sample Text" , (255 , 255 , 255 ), font = font )
132
+ draw .text ((4 , 4 ), time_str , (255 , 255 , 255 ), font = font )
112
133
113
134
bytes_img = BytesIO ()
114
- temp .save (bytes_img , 'png' )
115
- merged_frames .append (bytes_img .getvalue ())
135
+ temp .save (bytes_img , 'png' , compress_level = 8 )
136
+
137
+ sequence .append (
138
+ AnimationFrameData (
139
+ time = time_image ,
140
+ image = bytes_img .getvalue ()
141
+ )
142
+ )
143
+
144
+ if most_recent_frame is None and current_time < time_image :
145
+ recent_idx = idx - 1 if idx > 0 else idx
146
+ most_recent_frame = sequence [recent_idx ].get ('image' , None )
147
+ _LOGGER .debug (f"Most recent frame is at { sequence [recent_idx ].get ('time' )} " )
148
+
149
+ background .close ()
150
+ most_recent_frame = most_recent_frame if most_recent_frame is not None else sequence [- 1 ].get ('image' )
116
151
117
- return {'animation' : {
118
- 'images' : merged_frames ,
119
- # TODO support translation for hint
120
- 'hint' : api_data .get ('animation' , {}).get ('sequenceHint' , {}).get ('en' )
121
- }
122
- }
152
+ return RadarAnimationData (
153
+ sequence = sequence ,
154
+ most_recent_image = most_recent_frame
155
+ )
156
+
157
+ async def process_api_data (self , api_data : dict ) -> ProcessedCoordinatorData :
158
+
159
+ return ProcessedCoordinatorData (
160
+ current_weather = IrmKmiCoordinator .current_weather_from_data (api_data ),
161
+ daily_forecast = IrmKmiCoordinator .daily_list_to_forecast (api_data .get ('for' , {}).get ('daily' )),
162
+ hourly_forecast = IrmKmiCoordinator .hourly_list_to_forecast (api_data .get ('for' , {}).get ('hourly' )),
163
+ animation = await self ._async_animation_data (api_data = api_data )
164
+ )
123
165
124
166
@staticmethod
125
- def process_api_data (api_data ) :
167
+ def current_weather_from_data (api_data : dict ) -> CurrentWeatherData :
126
168
# Process data to get current hour forecast
127
169
now_hourly = None
128
170
hourly_forecast_data = api_data .get ('for' , {}).get ('hourly' )
@@ -140,23 +182,18 @@ def process_api_data(api_data):
140
182
for module in module_data :
141
183
if module .get ('type' , None ) == 'uv' :
142
184
uv_index = module .get ('data' , {}).get ('levelValue' )
143
- # Put everything together
185
+
144
186
# TODO NL cities have a better 'obs' section, use that for current weather
145
- processed_data = {
146
- 'current_weather' : {
147
- 'condition' : CDT_MAP .get (
148
- (api_data .get ('obs' , {}).get ('ww' ), api_data .get ('obs' , {}).get ('dayNight' )), None ),
149
- 'temperature' : api_data .get ('obs' , {}).get ('temp' ),
150
- 'wind_speed' : now_hourly .get ('windSpeedKm' , None ) if now_hourly is not None else None ,
151
- 'wind_gust_speed' : now_hourly .get ('windPeakSpeedKm' , None ) if now_hourly is not None else None ,
152
- 'wind_bearing' : now_hourly .get ('windDirectionText' , {}).get ('en' ) if now_hourly is not None else None ,
153
- 'pressure' : now_hourly .get ('pressure' , None ) if now_hourly is not None else None ,
154
- 'uv_index' : uv_index
155
- },
156
- 'daily_forecast' : IrmKmiCoordinator .daily_list_to_forecast (api_data .get ('for' , {}).get ('daily' )),
157
- 'hourly_forecast' : IrmKmiCoordinator .hourly_list_to_forecast (api_data .get ('for' , {}).get ('hourly' ))
158
- }
159
- return processed_data
187
+ current_weather = CurrentWeatherData (
188
+ condition = CDT_MAP .get ((api_data .get ('obs' , {}).get ('ww' ), api_data .get ('obs' , {}).get ('dayNight' )), None ),
189
+ temperature = api_data .get ('obs' , {}).get ('temp' ),
190
+ wind_speed = now_hourly .get ('windSpeedKm' , None ) if now_hourly is not None else None ,
191
+ wind_gust_speed = now_hourly .get ('windPeakSpeedKm' , None ) if now_hourly is not None else None ,
192
+ wind_bearing = now_hourly .get ('windDirectionText' , {}).get ('en' ) if now_hourly is not None else None ,
193
+ pressure = now_hourly .get ('pressure' , None ) if now_hourly is not None else None ,
194
+ uv_index = uv_index
195
+ )
196
+ return current_weather
160
197
161
198
@staticmethod
162
199
def hourly_list_to_forecast (data : List [dict ] | None ) -> List [Forecast ] | None :
0 commit comments