From 4ba1d94343559c4e7a1f8736dd2eb43049291b6e Mon Sep 17 00:00:00 2001 From: Ulrich Norbisrath Date: Tue, 14 Apr 2020 20:42:26 -0300 Subject: [PATCH 1/2] Re-add local pyown usage with local api-key. Refactored tOWNApi into real wrapper that can use local api_key or default to doing web requests to mycroft. --- __init__.py | 50 ++++++++++++++++++++++++++++++++++++-------------- 1 file changed, 36 insertions(+), 14 deletions(-) diff --git a/__init__.py b/__init__.py index 5a2766e3..e52bcfce 100644 --- a/__init__.py +++ b/__init__.py @@ -32,6 +32,7 @@ from pyowm.webapi25.forecaster import Forecaster from pyowm.webapi25.forecastparser import ForecastParser from pyowm.webapi25.observationparser import ObservationParser +from pyowm import OWM from requests import HTTPError, Response try: @@ -80,9 +81,10 @@ class LocationNotFoundError(ValueError): class OWMApi(Api): ''' Wrapper that defaults to the Mycroft cloud proxy so user's don't need - to get their own OWM API keys ''' + to get their own OWM API keys. + However it also handles a local OWM api_key.''' - def __init__(self): + def __init__(self, api_key, use_proxy): super(OWMApi, self).__init__("owm") self.owmlang = "en" self.encoding = "utf8" @@ -91,6 +93,12 @@ def __init__(self): self.query_cache = {} self.location_translations = {} + if api_key and not use_proxy: + self.owm = OWM(api_key) + else: + self.owm = None + + @staticmethod def get_language(lang): """ @@ -137,6 +145,7 @@ def build_query(self, params): def request(self, data): """ Caching the responses """ + # request is not used for local_api key req_hash = hash(json.dumps(data, sort_keys=True)) cache = self.query_cache.get(req_hash, (0, None)) # check for caches with more days data than requested @@ -170,6 +179,9 @@ def weather_at_location(self, name): if name == '': raise LocationNotFoundError('The location couldn\'t be found') + if self.owm is not None: + return self.owm.weather_at_location(name) + q = {"q": name} try: data = self.request({ @@ -185,10 +197,14 @@ def weather_at_location(self, name): def weather_at_place(self, name, lat, lon): if lat and lon: + if self.owm is not None: + return self.owm.weather_at_coords(lat, lon) q = {"lat": lat, "lon": lon} else: if name in self.location_translations: name = self.location_translations[name] + if self.owm is not None: + return self.owm.weather_at_place(name) response, trans_name = self.weather_at_location(name) self.location_translations[name] = trans_name return response @@ -201,10 +217,14 @@ def weather_at_place(self, name, lat, lon): def three_hours_forecast(self, name, lat, lon): if lat and lon: + if self.owm is not None: + return self.owm.three_hours_forecast_at_coords(lat, lon) q = {"lat": lat, "lon": lon} else: if name in self.location_translations: name = self.location_translations[name] + if self.owm is not None: + return self.owm.three_hours_forecast(name) q = {"q": name} data = self.request({ @@ -239,7 +259,11 @@ def _daily_forecast_at_location(self, name, limit): def daily_forecast(self, name, lat, lon, limit=None): if lat and lon: q = {"lat": lat, "lon": lon} + if self.owm is not None: + return self.owm.daily_forecast_at_coords(lat, lon) else: + if self.owm is not None: + return self.owm.daily_forecast(name, limit) return self._daily_forecast_at_location(name, limit) if limit is not None: @@ -268,6 +292,8 @@ def set_OWM_language(self, lang): 'se': 'latin1' } self.encoding = encodings.get(lang, 'utf8') + if self.owm is not None: + self.owm.set_language(lang) class WeatherSkill(MycroftSkill): @@ -288,23 +314,19 @@ def __init__(self): self.CODES['50d', '50n'] = 7 # windy/misty # Use Mycroft proxy if no private key provided - self.settings["api_key"] = None - self.settings["use_proxy"] = True + self.settings.setdefault("api_key", None) + self.settings.setdefault("use_proxy", True) def initialize(self): - # TODO: Remove lat,lon parameters from the OWMApi() - # methods and implement _at_coords() versions - # instead to make the interfaces compatible - # again. - # - # if self.settings["api_key"] and not self.settings['use_proxy']): - # self.owm = OWM(self.settings["api_key"]) - # else: - # self.owm = OWMApi() - self.owm = OWMApi() + # lat,lon parameters are handled by OWMApi() both for + # local api key and mycrodt's api + + self.owm = OWMApi( self.settings["api_key"], self.settings['use_proxy'] ) + if self.owm: self.owm.set_OWM_language(lang=OWMApi.get_language(self.lang)) + self.schedule_for_daily_use() try: self.mark2_forecast(self.__initialize_report(None)) From 1a0d5bf0afd267381fa9893b80af30182bd02ccb Mon Sep 17 00:00:00 2001 From: Ulrich Norbisrath Date: Mon, 20 Apr 2020 16:11:00 -0300 Subject: [PATCH 2/2] add to deal with city/state combos --- __init__.py | 17 +++++++++++++++-- 1 file changed, 15 insertions(+), 2 deletions(-) diff --git a/__init__.py b/__init__.py index e52bcfce..cd26f3f7 100644 --- a/__init__.py +++ b/__init__.py @@ -195,6 +195,16 @@ def weather_at_location(self, name): return self.weather_at_location(name) raise + def check_for_city_state(self, name): + # someone might ask for denver colorado (which would need to be written denver,colorado) + reg = self.owm.city_id_registry() + if len(reg.ids_for(name)) == 0: + # Try to rewrite name -> replace last blank with comma # TODO: careful this might be language specific -> check + p = name.rfind(" ") + if p>0: + name = name[:p] + "," + name[p+1:] + return name + def weather_at_place(self, name, lat, lon): if lat and lon: if self.owm is not None: @@ -203,7 +213,9 @@ def weather_at_place(self, name, lat, lon): else: if name in self.location_translations: name = self.location_translations[name] + LOG.info("searching weather for {}".format(name)) # TODO: remove debug if self.owm is not None: + name = self.check_for_city_state(name) return self.owm.weather_at_place(name) response, trans_name = self.weather_at_location(name) self.location_translations[name] = trans_name @@ -236,6 +248,9 @@ def three_hours_forecast(self, name, lat, lon): def _daily_forecast_at_location(self, name, limit): if name in self.location_translations: name = self.location_translations[name] + if self.owm is not None: + name = self.check_for_city_state(name) + return self.owm.daily_forecast(name, limit) orig_name = name while name != '': try: @@ -262,8 +277,6 @@ def daily_forecast(self, name, lat, lon, limit=None): if self.owm is not None: return self.owm.daily_forecast_at_coords(lat, lon) else: - if self.owm is not None: - return self.owm.daily_forecast(name, limit) return self._daily_forecast_at_location(name, limit) if limit is not None: