Source code for platypush.plugins.sun

import datetime
from typing import Dict, Optional

import requests
from dateutil.tz import gettz

from platypush.message.event.sun import SunriseEvent, SunsetEvent
from platypush.plugins import RunnablePlugin, action
from platypush.schemas.sun import SunEventsSchema
from platypush.utils import utcnow


[docs] class SunPlugin(RunnablePlugin): """ Plugin to get sunset/sunrise events and info for a certain location. """ _base_url = 'https://api.sunrise-sunset.org/json' _schema = SunEventsSchema() _attr_to_event_class = { 'sunrise': SunriseEvent, 'sunset': SunsetEvent, }
[docs] def __init__(self, latitude: float, longitude: float, **kwargs): """ :param latitude: Default latitude. :param longitude: Default longitude. """ super().__init__(**kwargs) self.latitude = latitude self.longitude = longitude
[docs] def main(self): while not self.should_stop(): next_events = self._get_events() next_event = next( iter( sorted( [ event_class( latitude=self.latitude, longitude=self.longitude, time=next_events[attr], ) for attr, event_class in self._attr_to_event_class.items() if next_events.get(attr) ], key=lambda t: t.time, ) ), None, ) assert next_event is not None, 'No next event found' wait_secs = max( 0, (next_event.time - datetime.datetime.now(tz=gettz())).seconds ) self.wait_stop(wait_secs) if not self.should_stop(): self._bus.post(next_event) self.wait_stop(2)
@staticmethod def _convert_time(t: str) -> datetime.datetime: now = utcnow().replace(microsecond=0) dt = datetime.datetime.strptime( f'{now.year}-{now.month:02d}-{now.day:02d} {t}', '%Y-%m-%d %I:%M:%S %p', ).replace(tzinfo=datetime.timezone.utc) if dt < now: dt += datetime.timedelta(days=1) return dt def _get_events( self, latitude: Optional[float] = None, longitude: Optional[float] = None ) -> Dict[str, datetime.datetime]: response = ( requests.get( self._base_url, timeout=10, params={ 'lat': latitude or self.latitude, 'lng': longitude or self.longitude, }, ) .json() .get('results', {}) ) return { attr: self._convert_time(t) for attr, t in response.items() if attr in self._schema.declared_fields }
[docs] @action def get_events( self, latitude: Optional[float] = None, longitude: Optional[float] = None ) -> dict: """ Return the next sun events. :param latitude: Override the default latitude. :param longitude: Override the default longitude. :return: .. schema:: sun.SunEventsSchema """ schema = SunEventsSchema() return dict( schema.dump(self._get_events(latitude=latitude, longitude=longitude)) )