Restructured files
Every command handler and related functions are now in seperate files Also made a seperate exception for not found weather station
This commit is contained in:
parent
b82dba3a9f
commit
7a24a127c2
4 changed files with 189 additions and 0 deletions
2
bot.py
2
bot.py
|
@ -6,6 +6,8 @@ from telegram import InlineQueryResultArticle, InputTextMessageContent, InlineKe
|
||||||
from telegram.ext import Updater, CommandHandler, InlineQueryHandler, CallbackQueryHandler
|
from telegram.ext import Updater, CommandHandler, InlineQueryHandler, CallbackQueryHandler
|
||||||
|
|
||||||
from exceptions import *
|
from exceptions import *
|
||||||
|
from vvs import inline_station_search, handle_vvs, handle_multiple_stations_reply
|
||||||
|
from weather_meteomedia import handle_weather
|
||||||
|
|
||||||
token_file = open("token", "r")
|
token_file = open("token", "r")
|
||||||
updater = Updater(token=token_file.read(), use_context=True)
|
updater = Updater(token=token_file.read(), use_context=True)
|
||||||
|
|
|
@ -6,6 +6,10 @@ class StationNotFoundError(Exception):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class WeatherStationNotFoundError(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
class ServerCommunicationError(Exception):
|
class ServerCommunicationError(Exception):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
@ -19,3 +23,19 @@ class MultipleStationsFoundError(Exception):
|
||||||
self.message_text = message_text
|
self.message_text = message_text
|
||||||
self.queried_station = queried_station
|
self.queried_station = queried_station
|
||||||
self.station_list = station_list
|
self.station_list = station_list
|
||||||
|
|
||||||
|
|
||||||
|
def error_callback(update, context):
|
||||||
|
try:
|
||||||
|
raise context.error
|
||||||
|
except NoArgError:
|
||||||
|
update.message.reply_text('No argument specified!')
|
||||||
|
return
|
||||||
|
except StationNotFoundError:
|
||||||
|
update.message.reply_text('No station matching this name found!')
|
||||||
|
return
|
||||||
|
except ServerCommunicationError:
|
||||||
|
update.message.reply_text('Error with server communication')
|
||||||
|
return
|
||||||
|
except WeatherStationNotFoundError:
|
||||||
|
update.message.reply_text('No weather station matching this name found!')
|
||||||
|
|
144
vvs.py
Normal file
144
vvs.py
Normal file
|
@ -0,0 +1,144 @@
|
||||||
|
import re
|
||||||
|
import requests
|
||||||
|
from telegram import InlineQueryResultArticle, InputTextMessageContent, InlineKeyboardButton, InlineKeyboardMarkup
|
||||||
|
|
||||||
|
from exceptions import MultipleStationsFoundError, NoArgError, ServerCommunicationError, StationNotFoundError
|
||||||
|
|
||||||
|
sent_multiple_station_message_ids = dict()
|
||||||
|
|
||||||
|
|
||||||
|
class Query:
|
||||||
|
station_id = -1
|
||||||
|
departure_count = 4
|
||||||
|
line = ''
|
||||||
|
destination = ""
|
||||||
|
|
||||||
|
def __init__(self, args):
|
||||||
|
request_tmp = ' '.join(args)
|
||||||
|
argument_names = re.findall(r' to | in | times ', request_tmp)
|
||||||
|
arguments = re.split(r' to | in | times ', request_tmp)
|
||||||
|
if not arguments[0].isdigit():
|
||||||
|
reply = search_vvs_station(arguments[0])
|
||||||
|
if len(reply) == 1:
|
||||||
|
self.station_id = reply[0]['stationId']
|
||||||
|
else:
|
||||||
|
raise MultipleStationsFoundError(request_tmp, arguments[0], reply)
|
||||||
|
else:
|
||||||
|
self.station_id = arguments[0]
|
||||||
|
|
||||||
|
if ' to ' in argument_names:
|
||||||
|
self.destination = arguments[argument_names.index(' to ') + 1]
|
||||||
|
if ' in ' in argument_names:
|
||||||
|
self.line = arguments[argument_names.index(' in ') + 1]
|
||||||
|
if ' times ' in argument_names:
|
||||||
|
self.departure_count = int(arguments[argument_names.index(' times ') + 1])
|
||||||
|
|
||||||
|
|
||||||
|
def build_menu(buttons,
|
||||||
|
n_cols,
|
||||||
|
header_buttons=None,
|
||||||
|
footer_buttons=None):
|
||||||
|
menu = [buttons[i:i + n_cols] for i in range(0, len(buttons), n_cols)]
|
||||||
|
if header_buttons:
|
||||||
|
menu.insert(0, [header_buttons])
|
||||||
|
if footer_buttons:
|
||||||
|
menu.append([footer_buttons])
|
||||||
|
return menu
|
||||||
|
|
||||||
|
|
||||||
|
def reply_multiple_stations(message, message_text, queried_station, station_list):
|
||||||
|
button_list = []
|
||||||
|
for station in station_list:
|
||||||
|
button_list.append(InlineKeyboardButton(station['fullName'],
|
||||||
|
callback_data="/vvs " + message_text
|
||||||
|
.replace(queried_station, station['stationId'])))
|
||||||
|
reply_markup = InlineKeyboardMarkup(build_menu(button_list, n_cols=2))
|
||||||
|
sent_multiple_station_message_ids[message.chat_id] = message.reply_text("Multiple stations found:",
|
||||||
|
reply_markup=reply_markup).message_id
|
||||||
|
|
||||||
|
|
||||||
|
def handle_multiple_stations_reply(update, context):
|
||||||
|
query = parse_station(update.callback_query.data.split(' ')[1:])
|
||||||
|
departures = get_vvs_departures(query)
|
||||||
|
for reply in departures:
|
||||||
|
context.bot.send_message(update.effective_chat['id'], reply)
|
||||||
|
context.bot.delete_message(chat_id=update.effective_chat['id'],
|
||||||
|
message_id=sent_multiple_station_message_ids[update.effective_chat['id']])
|
||||||
|
del sent_multiple_station_message_ids[update.effective_chat['id']]
|
||||||
|
|
||||||
|
|
||||||
|
def search_vvs_station(query):
|
||||||
|
request = requests.get("https://efa-api.asw.io/api/v1/station/?search=" + query)
|
||||||
|
if request.status_code != 200:
|
||||||
|
raise ServerCommunicationError
|
||||||
|
if request.status_code == 200 and len(request.text) <= 2:
|
||||||
|
raise StationNotFoundError
|
||||||
|
else:
|
||||||
|
return request.json()
|
||||||
|
|
||||||
|
|
||||||
|
def handle_vvs(update, context):
|
||||||
|
try:
|
||||||
|
query = parse_station(context.args)
|
||||||
|
except MultipleStationsFoundError as error:
|
||||||
|
reply_multiple_stations(update.message, error.message_text, error.queried_station, error.station_list)
|
||||||
|
return
|
||||||
|
departures = get_vvs_departures(query)
|
||||||
|
|
||||||
|
for reply in departures:
|
||||||
|
update.message.reply_text(reply, disable_notification=True)
|
||||||
|
|
||||||
|
|
||||||
|
def parse_station(args):
|
||||||
|
if len(args) == 0:
|
||||||
|
raise NoArgError
|
||||||
|
try:
|
||||||
|
query = Query(args)
|
||||||
|
except StationNotFoundError:
|
||||||
|
raise
|
||||||
|
except MultipleStationsFoundError:
|
||||||
|
raise
|
||||||
|
if query.station_id == -1:
|
||||||
|
raise StationNotFoundError
|
||||||
|
return query
|
||||||
|
|
||||||
|
|
||||||
|
def get_vvs_departures(query):
|
||||||
|
reply = []
|
||||||
|
request = requests.get("https://efa-api.asw.io/api/v1/station/" + query.station_id + "/departures")
|
||||||
|
if request.status_code != 200:
|
||||||
|
raise ServerCommunicationError
|
||||||
|
reply.append("Next departures for station: " + request.json()[0]['stopName'])
|
||||||
|
printed_departures = 0
|
||||||
|
for station in request.json():
|
||||||
|
if station['direction'].casefold().find(query.destination.casefold()) != -1:
|
||||||
|
if query.line == '' or (query.line != -1 and station['number'] == query.line):
|
||||||
|
reply.append(
|
||||||
|
"Line " + station['number'] + " to \"" + station['direction'] + "\" at "
|
||||||
|
+ station['departureTime']['hour'].zfill(2) + ':' + station['departureTime']['minute'].zfill(2)
|
||||||
|
+ " (+" + str(station['delay']) + ")")
|
||||||
|
printed_departures += 1
|
||||||
|
if printed_departures >= query.departure_count:
|
||||||
|
break
|
||||||
|
return reply
|
||||||
|
|
||||||
|
|
||||||
|
def inline_station_search(update, context):
|
||||||
|
query = update.inline_query.query
|
||||||
|
if len(query) < 5 or not query:
|
||||||
|
return
|
||||||
|
results = list()
|
||||||
|
for station in get_station_id_list(query):
|
||||||
|
results.append(
|
||||||
|
InlineQueryResultArticle(
|
||||||
|
id=station['stationId'],
|
||||||
|
title=station['name'],
|
||||||
|
input_message_content=InputTextMessageContent("/vvs " + station['stationId'])
|
||||||
|
)
|
||||||
|
)
|
||||||
|
context.bot.answer_inline_query(update.inline_query.id, results)
|
||||||
|
|
||||||
|
|
||||||
|
def get_station_id_list(name):
|
||||||
|
request = requests.get("https://efa-api.asw.io/api/v1/station/?search=" + name)
|
||||||
|
return request.json()
|
23
weather_meteomedia.py
Normal file
23
weather_meteomedia.py
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
from exceptions import NoArgError, WeatherStationNotFoundError
|
||||||
|
|
||||||
|
|
||||||
|
def search_weather_station(query: str):
|
||||||
|
station_file = open('stations', 'r')
|
||||||
|
for line in station_file:
|
||||||
|
if line.casefold().find(query.casefold()) != -1:
|
||||||
|
station_file.close()
|
||||||
|
return line
|
||||||
|
return 'none'
|
||||||
|
|
||||||
|
|
||||||
|
def handle_weather(update, context):
|
||||||
|
if len(context.args) == 0:
|
||||||
|
raise NoArgError
|
||||||
|
station_line = search_weather_station(' '.join(context.args))
|
||||||
|
if station_line == 'none':
|
||||||
|
raise WeatherStationNotFoundError
|
||||||
|
separator_index = station_line.find(' ')
|
||||||
|
update.message.reply_text('Weather for ' + station_line[separator_index + 1:-1]
|
||||||
|
+ ':\nhttp://wetterstationen.meteomedia.de/messnetz/vorhersagegrafik/' +
|
||||||
|
station_line[:separator_index] + '.png',
|
||||||
|
disable_notification=True)
|
Loading…
Reference in a new issue