Wetter und Forecast Plugin auf forecast.io (by @iCON)

This commit is contained in:
Akamaru 2016-02-16 18:53:15 +01:00
parent 8412921315
commit b1db1af73f
6 changed files with 393 additions and 109 deletions

View File

@ -210,6 +210,7 @@ function create_cred()
derpibooru_apikey = "", derpibooru_apikey = "",
fb_access_token = "", fb_access_token = "",
flickr_apikey = "", flickr_apikey = "",
forecastio_apikey = "",
ftp_site = "", ftp_site = "",
ftp_username = "", ftp_username = "",
ftp_password = "", ftp_password = "",

View File

@ -774,3 +774,9 @@ function is_blacklisted(msg)
end end
return var return var
end end
function convert_timestamp(timestamp, format)
local converted_date = run_command('date -d @'..timestamp..' +'..format)
local converted_date = string.gsub(converted_date, '%\n', '')
return converted_date
end

View File

@ -1,69 +1,92 @@
do do
require("./plugins/time")
local BASE_URL = "https://api.forecast.io/forecast"
local apikey = cred_data.forecastio_apikey
local google_apikey = cred_data.google_apikey
local function get_city_name(lat, lng)
local city = redis:hget('telegram:cache:weather:pretty_names', lat..','..lng)
if city then return city end
local url = 'https://maps.googleapis.com/maps/api/geocode/json?latlng='..lat..','..lng..'&result_type=political&language=de&key='..google_apikey
local res, code = https.request(url)
if code ~= 200 then return 'Unbekannte Stadt' end
local data = json:decode(res).results[1]
local city = data.formatted_address
print('Setting '..lat..','..lng..' in redis hash telegram:cache:weather:pretty_names to "'..city..'"')
redis:hset('telegram:cache:weather:pretty_names', lat..','..lng, city)
return city
end
local BASE_URL = "http://api.openweathermap.org/data/2.5/forecast/daily"
local function get_condition_symbol(weather, n) local function get_condition_symbol(weather, n)
if weather.list[n].weather[1].main == 'Clear' then if weather.data[n].icon == 'clear-day' then
return '' return '☀️'
elseif weather.list[n].weather[1].main == 'Clouds' then elseif weather.data[n].icon == 'clear-night' then
return ' ☁☁' return '🌙'
elseif weather.list[n].weather[1].main == 'Rain' then elseif weather.data[n].icon == 'rain' then
return '' return '☔️'
elseif weather.list[n].weather[1].main == 'Thunderstorm' then elseif weather.data[n].icon == 'snow' then
return ' ☔☔☔☔'
elseif weather.list[n].weather[1].main == 'Snow' then
return '❄️' return '❄️'
elseif weather.weather[1].main == 'Fog' then elseif weather.data[n].icon == 'sleet' then
return '🌨'
elseif weather.data[n].icon == 'wind' then
return '💨'
elseif weather.data[n].icon == 'fog' then
return '🌫' return '🌫'
elseif weather.data[n].icon == 'cloudy' then
return '☁️☁️'
elseif weather.data[n].icon == 'partly-cloudy-day' then
return '🌤'
elseif weather.data[n].icon == 'partly-cloudy-night' then
return '🌙☁️'
else else
return '' return ''
end end
end end
local function get_temp(weather, n) local function get_temp(weather, n)
local day = (round(weather.list[n].temp.day, 1)) local day = round(weather.data[n].temperatureMax, 1)
local night = (round(weather.list[n].temp.night, 1)) local night = round(weather.data[n].temperatureMin, 1)
local condition = weather.list[n].weather[1].description local condition = weather.data[n].summary
return '☀️ '..day..'°C | 🌙 '..night..'°C | '..condition return '☀️ '..day..'°C | 🌙 '..night..'°C | '..get_condition_symbol(weather, n)..' '..condition
end end
local function get_forecast(location, days) local function get_forecast(lat, lng)
print("Bekomme Wettervorhersage für ", location) print('Finde Wetter in '..lat..', '..lng)
local location = string.gsub(location," ","+") local text = redis:get('telegram:cache:forecast:'..lat..','..lng)
local url = BASE_URL if text then print('...aus dem Cache..') return text end
local apikey = cred_data.owm_apikey
local url = url..'?q='..location
local url = url..'&lang=de&units=metric&cnt='..days..'&APPID='..apikey
local b, c, h = http.request(url) local url = BASE_URL..'/'..apikey..'/'..lat..','..lng..'?lang=de&units=si&exclude=currently,minutely,hourly,alerts,flags'
if c ~= 200 then return nil end
local weather = json:decode(b) local response_body = {}
local city = weather.city.name local request_constructor = {
if weather.city.country == "" then url = url,
country = '' method = "GET",
else sink = ltn12.sink.table(response_body)
country = ' ('..weather.city.country..')' }
local ok, response_code, response_headers, response_status_line = https.request(request_constructor)
if not ok then return nil end
local data = json:decode(table.concat(response_body))
local ttl = string.sub(response_headers["cache-control"], 9)
local weather = data.daily
local city = get_city_name(lat, lng)
local header = 'Vorhersage für '..city..':\n'..weather.summary..'\n'
local text = 'Heute: '..get_temp(weather, 1)
local text = text..'\nMorgen: '..get_temp(weather, 2)
for day in pairs(weather.data) do
if day > 2 then
text = text..'\n'..convert_timestamp(weather.data[day].time, '%d.%m')..': '..get_temp(weather, day)
end end
local header = 'Vorhersage für '..city..country..':\n'
local text = 'Heute: '..get_temp(weather, 1)..get_condition_symbol(weather, 1)
if days > 1 then
text = text..'\nMorgen: '..get_temp(weather, 2)..get_condition_symbol(weather, 2)
end
if days > 2 then
text = text..'\nÜbermorgen: '..get_temp(weather, 3)..get_condition_symbol(weather, 3)
end end
if days > 3 then cache_data('forecast', lat..','..lng, header..text, tonumber(ttl), 'key')
for day in pairs(weather.list) do
if day > 3 then
local actual_day = day-1
text = text..'\n'..actual_day..' Tage: '..get_temp(weather, day)..get_condition_symbol(weather, day)
end
end
end
return header..text return header..text
end end
@ -71,28 +94,33 @@ end
local function run(msg, matches) local function run(msg, matches)
local user_id = msg.from.id local user_id = msg.from.id
local city = get_location(user_id) local city = get_location(user_id)
if not city then city = 'Berlin' end
if tonumber(matches[1]) then if matches[1] ~= '/forecast' and matches[1] ~= '/f' then
days = matches[1]+1
else
days = 4
end
if matches[2] then
days = matches[1]+1
city = matches[2]
end
if not tonumber(matches[1]) and matches[1] ~= '/forecast' then
city = matches[1] city = matches[1]
else
local set_location = get_location(user_id)
if not set_location then
city = 'Berlin, Deutschland'
else
city = set_location
end
end end
if days > 17 then local lat = redis:hget('telegram:cache:weather:'..string.lower(city), 'lat')
return 'Wettervorhersagen gehen nur von 1-16 Tagen!' local lng = redis:hget('telegram:cache:weather:'..string.lower(city), 'lng')
if not lat and not lng then
print('Koordinaten nicht eingespeichert, frage Google...')
lat,lng = get_latlong(city)
end end
local text = get_forecast(city, days) if not lat and not lng then
return 'Den Ort "'..city..'" gibt es nicht!'
end
redis:hset('telegram:cache:weather:'..string.lower(city), 'lat', lat)
redis:hset('telegram:cache:weather:'..string.lower(city), 'lng', lng)
local text = get_forecast(lat, lng)
if not text then if not text then
text = 'Konnte die Wettervorhersage für diese Stadt nicht bekommen.' text = 'Konnte die Wettervorhersage für diese Stadt nicht bekommen.'
end end
@ -102,15 +130,13 @@ end
return { return {
description = "Wettervorhersage für deinen oder einen gewählten Ort", description = "Wettervorhersage für deinen oder einen gewählten Ort",
usage = { usage = {
"/forecast: Wettervorhersage für deine Stadt (!location set [Ort])", "/forecast: Wettervorhersage für deine Stadt (/location set [Ort])",
"/forecast [0-16]: Wettervorhersage für X Tage für deine Stadt (!location set [Ort])", "/forecast (Stadt): Wettervorhersage für diese Stadt"
"/forecast (Stadt): Wettervorhersage für diese Stadt",
"/forecast [0-16] (Stadt): Wettervorhersage für X Tage für diese Stadt"
}, },
patterns = { patterns = {
"^/f$",
"^/f (.*)$",
"^/forecast$", "^/forecast$",
"^/forecast (%d+) (.*)$",
"^/forecast (%d+)",
"^/forecast (.*)$" "^/forecast (.*)$"
}, },
run = run run = run

View File

@ -0,0 +1,119 @@
do
local BASE_URL = "http://api.openweathermap.org/data/2.5/forecast/daily"
local function get_condition_symbol(weather, n)
if weather.list[n].weather[1].main == 'Clear' then
return ''
elseif weather.list[n].weather[1].main == 'Clouds' then
return ' ☁☁'
elseif weather.list[n].weather[1].main == 'Rain' then
return ''
elseif weather.list[n].weather[1].main == 'Thunderstorm' then
return ' ☔☔☔☔'
elseif weather.list[n].weather[1].main == 'Snow' then
return ' ❄️'
elseif weather.weather[1].main == 'Fog' then
return ' 🌫'
else
return ''
end
end
local function get_temp(weather, n)
local day = (round(weather.list[n].temp.day, 1))
local night = (round(weather.list[n].temp.night, 1))
local condition = weather.list[n].weather[1].description
return '☀️ '..day..'°C | 🌙 '..night..'°C | '..condition
end
local function get_forecast(location, days)
print("Bekomme Wettervorhersage für ", location)
local location = string.gsub(location," ","+")
local url = BASE_URL
local apikey = cred_data.owm_apikey
local url = url..'?q='..location
local url = url..'&lang=de&units=metric&cnt='..days..'&APPID='..apikey
local b, c, h = http.request(url)
if c ~= 200 then return nil end
local weather = json:decode(b)
local city = weather.city.name
if weather.city.country == "" then
country = ''
else
country = ' ('..weather.city.country..')'
end
local header = 'Vorhersage für '..city..country..':\n'
local text = 'Heute: '..get_temp(weather, 1)..get_condition_symbol(weather, 1)
if days > 1 then
text = text..'\nMorgen: '..get_temp(weather, 2)..get_condition_symbol(weather, 2)
end
if days > 2 then
text = text..'\nÜbermorgen: '..get_temp(weather, 3)..get_condition_symbol(weather, 3)
end
if days > 3 then
for day in pairs(weather.list) do
if day > 3 then
local actual_day = day-1
text = text..'\n'..actual_day..' Tage: '..get_temp(weather, day)..get_condition_symbol(weather, day)
end
end
end
return header..text
end
local function run(msg, matches)
local user_id = msg.from.id
local city = get_location(user_id)
if not city then city = 'Berlin' end
if tonumber(matches[1]) then
days = matches[1]+1
else
days = 4
end
if matches[2] then
days = matches[1]+1
city = matches[2]
end
if not tonumber(matches[1]) and matches[1] ~= '/forecast' then
city = matches[1]
end
if days > 17 then
return 'Wettervorhersagen gehen nur von 1-16 Tagen!'
end
local text = get_forecast(city, days)
if not text then
text = 'Konnte die Wettervorhersage für diese Stadt nicht bekommen.'
end
return text
end
return {
description = "Wettervorhersage für deinen oder einen gewählten Ort",
usage = {
"/forecast: Wettervorhersage für deine Stadt (!location set [Ort])",
"/forecast [0-16]: Wettervorhersage für X Tage für deine Stadt (!location set [Ort])",
"/forecast (Stadt): Wettervorhersage für diese Stadt",
"/forecast [0-16] (Stadt): Wettervorhersage für X Tage für diese Stadt"
},
patterns = {
"^/forecast$",
"^/forecast (%d+) (.*)$",
"^/forecast (%d+)",
"^/forecast (.*)$"
},
run = run
}
end

View File

@ -0,0 +1,77 @@
do
local BASE_URL = "http://api.openweathermap.org/data/2.5/weather"
local function get_weather(location)
print("Finde Wetter in ", location)
local location = string.gsub(location," ","+")
local url = BASE_URL
local apikey = cred_data.owm_apikey
local url = url..'?q='..location
local url = url..'&lang=de&units=metric&APPID='..apikey
local b, c, h = http.request(url)
if c ~= 200 then return nil end
local weather = json:decode(b)
local city = weather.name
if weather.sys.country == 'none' then
country = ''
else
country = ' ('..weather.sys.country..')'
end
local temperature = round(weather.main.temp, 1)
local temp = 'Wetter in '..city..country..':\n'..temperature..'°C'
local conditions = ' | '..weather.weather[1].description
if weather.weather[1].main == 'Clear' then
conditions = conditions..''
elseif weather.weather[1].main == 'Clouds' then
conditions = conditions..' ☁☁'
elseif weather.weather[1].main == 'Rain' then
conditions = conditions..''
elseif weather.weather[1].main == 'Thunderstorm' then
conditions = conditions..' ☔☔☔☔'
elseif weather.weather[1].main == 'Snow' then
conditions = conditions..' ❄️'
elseif weather.weather[1].main == 'Fog' then
conditions = conditions..' 🌫'
else
conditions = conditions..''
end
return temp..conditions
end
local function run(msg, matches)
local user_id = msg.from.id
if matches[1] ~= '/wetter' then
city = matches[1]
else
local set_location = get_location(user_id)
if not set_location then
city = 'Berlin'
else
city = set_location
end
end
local text = get_weather(city)
if not text then
text = 'Konnte das Wetter von dieser Stadt nicht bekommen.'
end
return text
end
return {
description = "Wetter für deinen oder einen gewählten Ort",
usage = {
"/wetter: Wetter für deinen Wohnort (!location set [Ort])",
"/wetter (Stadt): Wetter für diese Stadt"
},
patterns = {
"^/wetter$",
"^/wetter (.*)$"
},
run = run
}
end

View File

@ -1,60 +1,113 @@
do do
local BASE_URL = "http://api.openweathermap.org/data/2.5/weather" require("./plugins/time")
local function get_weather(location) local BASE_URL = "https://api.forecast.io/forecast"
print("Finde Wetter in ", location) local apikey = cred_data.forecastio_apikey
local location = string.gsub(location," ","+") local google_apikey = cred_data.google_apikey
local url = BASE_URL
local apikey = cred_data.owm_apikey
local url = url..'?q='..location
local url = url..'&lang=de&units=metric&APPID='..apikey
local b, c, h = http.request(url) local function get_city_name(lat, lng)
if c ~= 200 then return nil end local city = redis:hget('telegram:cache:weather:pretty_names', lat..','..lng)
if city then return city end
local weather = json:decode(b) local url = 'https://maps.googleapis.com/maps/api/geocode/json?latlng='..lat..','..lng..'&result_type=political&language=de&key='..google_apikey
local city = weather.name local res, code = https.request(url)
if weather.sys.country == 'none' then if code ~= 200 then return 'Unbekannte Stadt' end
country = '' local data = json:decode(res).results[1]
else local city = data.formatted_address
country = ' ('..weather.sys.country..')' print('Setting '..lat..','..lng..' in redis hash telegram:cache:weather:pretty_names to "'..city..'"')
redis:hset('telegram:cache:weather:pretty_names', lat..','..lng, city)
return city
end end
local temperature = round(weather.main.temp, 1)
local temp = 'Wetter in '..city..country..':\n'..temperature..'°C' local function get_weather(lat, lng)
local conditions = ' | '..weather.weather[1].description print('Finde Wetter in '..lat..', '..lng)
if weather.weather[1].main == 'Clear' then local text = redis:get('telegram:cache:weather:'..lat..','..lng)
conditions = conditions..'' if text then print('...aus dem Cache..') return text end
elseif weather.weather[1].main == 'Clouds' then
conditions = conditions..' ☁☁' local url = BASE_URL..'/'..apikey..'/'..lat..','..lng..'?lang=de&units=si&exclude=minutely,hourly,daily,alerts,flags'
elseif weather.weather[1].main == 'Rain' then
conditions = conditions..'' local response_body = {}
elseif weather.weather[1].main == 'Thunderstorm' then local request_constructor = {
conditions = conditions..' ☔☔☔☔' url = url,
elseif weather.weather[1].main == 'Snow' then method = "GET",
sink = ltn12.sink.table(response_body)
}
local ok, response_code, response_headers, response_status_line = https.request(request_constructor)
if not ok then return nil end
local data = json:decode(table.concat(response_body))
local ttl = string.sub(response_headers["cache-control"], 9)
local weather = data.currently
local city = get_city_name(lat, lng)
local temperature = round(weather.temperature, 1)
local feelslike = round(weather.apparentTemperature, 1)
local temp = 'Wetter in '..city..':\n'..temperature..' °C'
local conditions = ' | '..weather.summary
if weather.icon == 'clear-day' then
conditions = conditions..' ☀️'
elseif weather.icon == 'clear-night' then
conditions = conditions..' 🌙'
elseif weather.icon == 'rain' then
conditions = conditions..' ☔️'
elseif weather.icon == 'snow' then
conditions = conditions..' ❄️' conditions = conditions..' ❄️'
elseif weather.weather[1].main == 'Fog' then elseif weather.icon == 'sleet' then
conditions = conditions..' 🌨'
elseif weather.icon == 'wind' then
conditions = conditions..' 💨'
elseif weather.icon == 'fog' then
conditions = conditions..' 🌫' conditions = conditions..' 🌫'
elseif weather.icon == 'cloudy' then
conditions = conditions..' ☁️☁️'
elseif weather.icon == 'partly-cloudy-day' then
conditions = conditions..' 🌤'
elseif weather.icon == 'partly-cloudy-night' then
conditions = conditions..' 🌙☁️'
else else
conditions = conditions..'' conditions = conditions..''
end end
return temp..conditions local windspeed = ' | 💨 '..round(weather.windSpeed, 1)..' m/s'
local text = temp..conditions..windspeed
if temperature == feelslike then
text = text..'\n(gefühlt: '..feelslike..' °C)'
end
cache_data('weather', lat..','..lng, text, tonumber(ttl), 'key')
return text
end end
local function run(msg, matches) local function run(msg, matches)
local user_id = msg.from.id local user_id = msg.from.id
if matches[1] ~= '/wetter' then if matches[1] ~= '/wetter' and matches[1] ~= '/w' then
city = matches[1] city = matches[1]
else else
local set_location = get_location(user_id) local set_location = get_location(user_id)
if not set_location then if not set_location then
city = 'Berlin' city = 'Berlin, Deutschland'
else else
city = set_location city = set_location
end end
end end
local text = get_weather(city)
local lat = redis:hget('telegram:cache:weather:'..string.lower(city), 'lat')
local lng = redis:hget('telegram:cache:weather:'..string.lower(city), 'lng')
if not lat and not lng then
print('Koordinaten nicht eingespeichert, frage Google...')
lat,lng = get_latlong(city)
end
if not lat and not lng then
return 'Den Ort "'..city..'" gibt es nicht!'
end
redis:hset('telegram:cache:weather:'..string.lower(city), 'lat', lat)
redis:hset('telegram:cache:weather:'..string.lower(city), 'lng', lng)
local text = get_weather(lat, lng)
if not text then if not text then
text = 'Konnte das Wetter von dieser Stadt nicht bekommen.' text = 'Konnte das Wetter von dieser Stadt nicht bekommen.'
end end
@ -64,12 +117,14 @@ end
return { return {
description = "Wetter für deinen oder einen gewählten Ort", description = "Wetter für deinen oder einen gewählten Ort",
usage = { usage = {
"/wetter: Wetter für deinen Wohnort (!location set [Ort])", "/wetter: Wetter für deinen Wohnort (/location set [Ort])",
"/wetter (Stadt): Wetter für diese Stadt" "/wetter (Stadt): Wetter für diese Stadt"
}, },
patterns = { patterns = {
"^/wetter$", "^/wetter$",
"^/wetter (.*)$" "^/wetter (.*)$",
"^/w$",
"^/w (.*)$"
}, },
run = run run = run
} }