feat: Added notes creation, view and deletion, added route planning, added location list with current weather and nearby locations, code improvements and fixes

This commit is contained in:
ITQ
2024-03-26 07:49:50 +03:00
parent 65719a61ef
commit 88dfe1704d
24 changed files with 1571 additions and 301 deletions
+600
View File
@@ -0,0 +1,600 @@
__all__ = ("router",)
import datetime
from aiogram import F, Router
from aiogram.exceptions import TelegramBadRequest
from aiogram.filters import StateFilter
from aiogram.fsm.context import FSMContext
from aiogram.types import CallbackQuery, Message
import sqlalchemy as sa
from app import messages, session
from app.config import Config
from app.filters.user import Registered, RegisteredCallback
from app.keyboards.builders import locations_keyboard, sights_keyboard
from app.keyboards.confirm_location import get as confirm_location_get
from app.keyboards.location import get as location_get
from app.models.travel import Location, Travel
from app.states.travel import (
CreateLocationState,
)
from app.utils.geo import get_location_by_name
from app.utils.sights import find_trips, get_info_by_xid
from app.utils.states import delete_message_from_state, handle_validation_error
from app.utils.weather import get_current_weather
router = Router(name="location_callback")
@router.callback_query(
F.data.startswith("travel_add_location"),
RegisteredCallback(),
StateFilter(None),
)
async def add_travel_location_callback(
callback: CallbackQuery,
state: FSMContext,
) -> None:
if (
callback.data is None
or callback.message is None
or not isinstance(callback.message, Message)
):
return
travel_id = int(callback.data.replace("travel_add_location_", ""))
travel = Travel().get_travel_by_id(travel_id)
if not travel:
return
await state.update_data(travel_id=travel_id)
await state.set_state(CreateLocationState.temp_location)
await callback.message.answer(
messages.CREATE_LOCATION,
)
await callback.message.answer(
messages.ENTER_LOCATION,
)
await callback.answer()
@router.message(CreateLocationState.temp_location, F.text, Registered())
async def location_entered(message: Message, state: FSMContext) -> None:
if (
message.text is None
or message.from_user is None
or message.bot is None
):
return
location = message.text.strip()
if location == "/cancel":
await message.answer(
messages.ACTION_CANCELED,
)
await state.update_data()
await message.delete()
await delete_message_from_state(
state,
message.chat.id,
message.bot,
)
await state.clear()
return
try:
validated_location = Location().validate_location(
key="location",
value=location,
)
except AssertionError as e:
await handle_validation_error(message, state, e)
return
await delete_message_from_state(state, message.chat.id, message.bot)
await state.update_data(
temp_location=validated_location,
temp_location_message_id=message.message_id,
)
await state.set_state(CreateLocationState.location)
await message.answer(
messages.CONFIRM_LOCATION.format(location=validated_location),
reply_markup=confirm_location_get(),
)
@router.callback_query(
F.data.in_(["confirm_location", "cancel_location"]),
RegisteredCallback(),
StateFilter(CreateLocationState.location),
)
async def confirm_location(
callback: CallbackQuery,
state: FSMContext,
) -> None:
if (
not callback.message
or not isinstance(callback.message, Message)
or callback.bot is None
):
return
data = await state.get_data()
location = data.get("temp_location")
if callback.data == "confirm_location":
await delete_message_from_state(
state,
callback.from_user.id,
callback.bot,
)
await state.update_data(location=location)
await state.set_state(CreateLocationState.date_start)
await callback.message.answer(
messages.INPUT_TRAVEL_CALLBACK.format(
key="location",
value=location,
),
)
await callback.message.answer(
messages.ENTER_LOCATION_DATE_START,
)
elif callback.data == "cancel_location":
error_message = await callback.message.answer(
messages.CONFIRMATION_REEJECTED,
)
try:
await callback.bot.delete_message(
callback.from_user.id,
data["temp_location_message_id"],
)
except TelegramBadRequest:
pass
await state.set_state(CreateLocationState.temp_location)
await state.update_data(error_message_id=error_message.message_id)
try:
await callback.message.delete()
except TelegramBadRequest:
pass
await callback.answer()
@router.message(CreateLocationState.date_start, F.text, Registered())
async def location_date_start_entered(
message: Message,
state: FSMContext,
) -> None:
if (
message.text is None
or message.from_user is None
or message.bot is None
):
return
date_start = message.text.strip()
if date_start == "/cancel":
await message.answer(
messages.ACTION_CANCELED,
)
await state.update_data()
await message.delete()
await delete_message_from_state(
state,
message.chat.id,
message.bot,
)
await state.clear()
return
try:
validated_date_start = Location().validate_date_end(
key="date_start",
value=date_start,
)
except AssertionError as e:
await handle_validation_error(message, state, e)
return
await delete_message_from_state(state, message.chat.id, message.bot)
await state.update_data(
date_start=datetime.datetime.strftime(
validated_date_start,
"%Y-%m-%d %H:%M:%S",
),
)
await state.set_state(CreateLocationState.date_end)
await message.answer(
messages.INPUT_TRAVEL_CALLBACK.format(
key="start date",
value=date_start,
),
)
await message.answer(
messages.ENTER_LOCATION_DATE_END,
)
@router.message(CreateLocationState.date_end, F.text, Registered())
async def location_date_end_entered(
message: Message,
state: FSMContext,
) -> None:
if (
message.text is None
or message.from_user is None
or message.bot is None
):
return
date_end = message.text.strip()
if date_end == "/cancel":
await message.answer(
messages.ACTION_CANCELED,
)
await state.update_data()
await message.delete()
await delete_message_from_state(
state,
message.chat.id,
message.bot,
)
await state.clear()
return
try:
validated_date_end = Location().validate_date_end(
key="date_end",
value=date_end,
)
except AssertionError as e:
await handle_validation_error(message, state, e)
return
date_start = (await state.get_data()).get("date_start")
if validated_date_end <= datetime.datetime.strptime(
str(date_start),
"%Y-%m-%d %H:%M:%S",
).replace(tzinfo=datetime.UTC):
await handle_validation_error(
message,
state,
messages.INVALID_DATE_END,
)
return
await state.update_data(
date_end=datetime.datetime.strftime(
validated_date_end,
"%Y-%m-%d %H:%M:%S",
),
)
data = await state.get_data()
overlapping_location = (
session.query(Location)
.filter(
sa.and_(
Location.travel_id == data["travel_id"],
Location.date_start < data["date_end"],
Location.date_end > data["date_start"],
),
)
.first()
)
if overlapping_location:
await handle_validation_error(
message,
state,
messages.OVERLAPPING_LOCATION,
)
return
await message.answer(
messages.INPUT_TRAVEL_CALLBACK.format(
key="end date",
value=date_end,
),
)
await delete_message_from_state(state, message.chat.id, message.bot)
if "temp_location" in data:
del data["temp_location"]
if "temp_location_message_id" in data:
del data["temp_location_message_id"]
if "error_message_id" in data:
del data["error_message_id"]
data["date_start"] = datetime.datetime.strptime(
data["date_start"],
"%Y-%m-%d %H:%M:%S",
)
data["date_end"] = datetime.datetime.strptime(
data["date_end"],
"%Y-%m-%d %H:%M:%S",
)
session.add(Location(**data))
session.commit()
await message.answer(
messages.LOCATION_ADDED,
)
await state.clear()
@router.callback_query(
F.data.startswith("travel_locations_page"),
RegisteredCallback(),
StateFilter(None),
)
async def travel_locations_page_callback(
callback: CallbackQuery,
):
if (
callback.message is None
or callback.data is None
or not isinstance(callback.message, Message)
):
return
travel_id, page = map(
int,
callback.data.replace("travel_locations_page_", "").split("_"),
)
travel = Travel.get_travel_by_id(travel_id)
if not travel or travel == []:
return
locations = Travel.get_sorted_locations(travel)
pages = (len(locations) + Config.PAGE_SIZE - 1) // Config.PAGE_SIZE
try:
await callback.message.edit_text(
messages.LOCATIONS,
reply_markup=locations_keyboard(
locations,
page,
pages,
travel_id,
),
)
except TelegramBadRequest:
pass
@router.callback_query(
F.data.startswith("travel_location_detail"),
RegisteredCallback(),
StateFilter(None),
)
async def travel_detail_location_callback(
callback: CallbackQuery,
):
if (
callback.message is None
or callback.data is None
or not isinstance(callback.message, Message)
):
return
location_id = int(callback.data.replace("travel_location_detail_", ""))
location = Location.get_location_by_id(location_id)
if not location or location == []:
return
try:
await callback.message.edit_text(
location.get_location_text(),
reply_markup=location_get(location.travel.id, location.id),
)
except TelegramBadRequest:
pass
@router.callback_query(
F.data.startswith("travel_locationdelete"),
RegisteredCallback(),
StateFilter(None),
)
async def travel_locations_delete_callback(
callback: CallbackQuery,
):
if (
callback.message is None
or callback.data is None
or not isinstance(callback.message, Message)
):
return
location_id = int(callback.data.replace("travel_locationdelete_", ""))
location_queryset = Location.get_location_queryset_by_id(location_id)
if not location_queryset or location_queryset == []:
return
travel = location_queryset.first().travel
location_queryset.delete()
session.commit()
locations = Travel.get_sorted_locations(travel)
pages = (len(locations) + Config.PAGE_SIZE - 1) // Config.PAGE_SIZE
try:
await callback.message.edit_text(
messages.LOCATIONS,
reply_markup=locations_keyboard(
locations,
0,
pages,
travel.id,
),
)
except TelegramBadRequest:
pass
await callback.message.answer(
messages.LOCATION_DELETED,
)
await callback.answer()
@router.callback_query(
F.data.startswith("travel_locationsights"),
RegisteredCallback(),
StateFilter(None),
)
async def travel_locationsights_callback(
callback: CallbackQuery,
):
if (
callback.message is None
or callback.data is None
or not isinstance(callback.message, Message)
):
return
location_id = int(callback.data.replace("travel_locationsights_", ""))
location = Location.get_location_by_id(location_id)
if not location or location == []:
return
geocode = get_location_by_name(location.location)
sights = find_trips(geocode[1].raw.get("lat"), geocode[1].raw.get("lon"))
if sights is None or len(sights) == 0:
await callback.message.answer(
messages.NO_SIGHTS_FOUND.format(
location=location.location,
distance=Config.NEARBY_SIGHTS_RADIUS,
),
)
else:
await callback.message.answer(
messages.SIGHTS_HEADER
+ messages.SIGHTS_FOOTER.format(
location=location.location,
sights_count=len(sights),
distance=Config.NEARBY_SIGHTS_RADIUS,
),
reply_markup=sights_keyboard(sights[:20]),
)
await callback.answer()
@router.callback_query(
F.data.startswith("travel_sight_detail"),
RegisteredCallback(),
StateFilter(None),
)
async def travel_sight_detail_callback(
callback: CallbackQuery,
):
if (
callback.message is None
or callback.data is None
or not isinstance(callback.message, Message)
):
return
sight_xid = callback.data.replace("travel_sight_detail_", "")
await get_info_by_xid(callback, sight_xid)
await callback.answer()
@router.callback_query(
F.data.startswith("travel_locationweather"),
RegisteredCallback(),
StateFilter(None),
)
async def travel_locationweather_callback(
callback: CallbackQuery,
):
if (
callback.message is None
or callback.data is None
or not isinstance(callback.message, Message)
):
return
location_id = int(callback.data.replace("travel_locationweather_", ""))
location = Location.get_location_by_id(location_id)
if not location or location == []:
return
geocode = get_location_by_name(location.location)
weather = get_current_weather(
geocode[1].raw.get("lat"),
geocode[1].raw.get("lon"),
)
await callback.message.answer(
messages.LOCATION_WEATHER.format(
location=location.location,
weather_main=weather.get("weather")[0].get("main"),
temp=weather.get("main").get("temp"),
feels_like=weather.get("main").get("feels_like"),
temp_min=weather.get("main").get("temp_min"),
temp_max=weather.get("main").get("temp_max"),
pressure=weather.get("main").get("pressure"),
humidity=weather.get("main").get("humidity"),
),
reply_to_message_id=callback.message.message_id,
)
await callback.answer()