Refactoring for reading the config
This commit is contained in:
parent
963ee6b86f
commit
fc11a1e1eb
9 changed files with 264 additions and 166 deletions
|
@ -4,30 +4,36 @@ import logging
|
||||||
import config
|
import config
|
||||||
from aiohttp import ClientSession
|
from aiohttp import ClientSession
|
||||||
from gestion_sports.gestion_sports_connector import GestionSportsConnector
|
from gestion_sports.gestion_sports_connector import GestionSportsConnector
|
||||||
from models import BookingFilter, User
|
from models import BookingFilter, Club, User
|
||||||
|
|
||||||
LOGGER = logging.getLogger(__name__)
|
LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
async def book(url: str, user: User, booking_filter: BookingFilter) -> None:
|
async def book(club: Club, user: User, booking_filter: BookingFilter) -> int | None:
|
||||||
async with ClientSession() as session:
|
async with ClientSession() as session:
|
||||||
platform = GestionSportsConnector(session, url)
|
platform = GestionSportsConnector(session, club.url)
|
||||||
await platform.connect()
|
await platform.connect()
|
||||||
await platform.login(user)
|
await platform.login(user, club)
|
||||||
await platform.book(booking_filter)
|
return await platform.book(booking_filter, club)
|
||||||
|
|
||||||
|
|
||||||
def main() -> None:
|
def main() -> int | None:
|
||||||
LOGGER.info("Starting booking padel court")
|
user = config.get_user()
|
||||||
|
booking_filter = config.get_booking_filter()
|
||||||
|
club = config.get_club()
|
||||||
|
|
||||||
LOGGER.info(
|
LOGGER.info(
|
||||||
f"login={config.USER}, password={config.PASSWORD}, club_id={config.CLUB_ID}"
|
"Starting booking court of %s for user %s at club %s at %s",
|
||||||
|
booking_filter.sport_id,
|
||||||
|
user.login,
|
||||||
|
club.id,
|
||||||
|
booking_filter.date,
|
||||||
)
|
)
|
||||||
user = User(login=config.USER, password=config.PASSWORD, club_id=config.CLUB_ID)
|
court_booked = asyncio.run(book(club, user, booking_filter))
|
||||||
LOGGER.info(
|
if court_booked:
|
||||||
f"court_id={config.COURT_IDS},sport_id={config.SPORT_ID},date={config.DATE_TIME}"
|
LOGGER.info(
|
||||||
)
|
"Court %s booked successfully at %s", court_booked, booking_filter.date
|
||||||
booking_filter = BookingFilter(
|
)
|
||||||
court_ids=config.COURT_IDS, sport_id=config.SPORT_ID, date=config.DATE_TIME
|
else:
|
||||||
)
|
LOGGER.info("Booking did not work")
|
||||||
asyncio.run(book(config.GESTION_SPORTS_URL, user, booking_filter))
|
return court_booked
|
||||||
LOGGER.info("Finished booking padel court")
|
|
||||||
|
|
|
@ -1,21 +1,50 @@
|
||||||
|
import json
|
||||||
import logging.config
|
import logging.config
|
||||||
import os
|
import os
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
import pendulum
|
import pendulum
|
||||||
import yaml
|
import yaml
|
||||||
from dotenv import load_dotenv
|
from dotenv import load_dotenv
|
||||||
|
|
||||||
|
from resa_padel.models import BookingFilter, Club, User
|
||||||
|
|
||||||
load_dotenv()
|
load_dotenv()
|
||||||
|
|
||||||
|
|
||||||
GESTION_SPORTS_URL = "https://toulousepadelclub.gestion-sports.com"
|
def get_club() -> Club:
|
||||||
USER = os.environ.get("USER")
|
club_url = os.environ.get("CLUB_URL")
|
||||||
PASSWORD = os.environ.get("PASSWORD")
|
court_ids_tmp = os.environ.get("COURT_IDS") or ""
|
||||||
CLUB_ID = os.environ.get("CLUB_ID")
|
court_ids = (
|
||||||
_court_ids = os.environ.get("COURT_IDS") or ""
|
[int(court_id) for court_id in court_ids_tmp.split(",")]
|
||||||
COURT_IDS = [int(court_id) for court_id in _court_ids.split(",")] if _court_ids else []
|
if court_ids_tmp
|
||||||
SPORT_ID = int(os.environ.get("SPORT_ID"))
|
else []
|
||||||
DATE_TIME = pendulum.parse(os.environ.get("DATE_TIME"))
|
)
|
||||||
|
club_id = os.environ.get("CLUB_ID")
|
||||||
|
return Club(id=club_id, url=club_url, courts_ids=court_ids)
|
||||||
|
|
||||||
|
|
||||||
|
def get_booking_filter() -> BookingFilter:
|
||||||
|
sport_id_tmp = os.environ.get("SPORT_ID")
|
||||||
|
sport_id = int(sport_id_tmp) if sport_id_tmp else None
|
||||||
|
date_time_tmp = os.environ.get("DATE_TIME")
|
||||||
|
date_time = pendulum.parse(date_time_tmp) if date_time_tmp else None
|
||||||
|
return BookingFilter(sport_id=sport_id, date=date_time)
|
||||||
|
|
||||||
|
|
||||||
|
def get_user() -> User:
|
||||||
|
login = os.environ.get("LOGIN")
|
||||||
|
password = os.environ.get("PASSWORD")
|
||||||
|
return User(login=login, password=password)
|
||||||
|
|
||||||
|
|
||||||
|
def get_post_headers(platform_id: str) -> dict:
|
||||||
|
root_path = Path(__file__).parent
|
||||||
|
headers_file = Path(root_path, "resources", platform_id, "post-headers.json")
|
||||||
|
with headers_file.open(mode="r", encoding="utf-8") as f:
|
||||||
|
headers = json.load(f)
|
||||||
|
|
||||||
|
return headers
|
||||||
|
|
||||||
|
|
||||||
def init_log_config():
|
def init_log_config():
|
||||||
|
|
|
@ -1,25 +1,20 @@
|
||||||
import asyncio
|
import asyncio
|
||||||
import json
|
import json
|
||||||
import logging
|
import logging
|
||||||
|
from urllib.parse import urljoin
|
||||||
|
|
||||||
from aiohttp import ClientResponse, ClientSession
|
from aiohttp import ClientResponse, ClientSession
|
||||||
from gestion_sports.gestion_sports_payload_builder import \
|
|
||||||
GestionSportsPayloadBuilder
|
import config
|
||||||
from models import BookingFilter, User
|
from gestion_sports.gestion_sports_payload_builder import GestionSportsPayloadBuilder
|
||||||
|
from models import BookingFilter, Club, User
|
||||||
|
|
||||||
|
DATE_FORMAT = "%d/%m/%Y"
|
||||||
|
|
||||||
|
TIME_FORMAT = "%H:%M"
|
||||||
|
|
||||||
LOGGER = logging.getLogger(__name__)
|
LOGGER = logging.getLogger(__name__)
|
||||||
HEADERS = {
|
POST_HEADERS = config.get_post_headers("gestion-sports")
|
||||||
"Connection": "keep-alive",
|
|
||||||
"Accept-Language": "en-US,en;q=0.5",
|
|
||||||
"Accept-Encoding": "gzip, deflate, br",
|
|
||||||
"DNT": "1",
|
|
||||||
"Content-Type": "application/x-www-form-urlencoded; charset=UTF-8",
|
|
||||||
"Accept": "application/json, text/javascript, */*; q=0.01",
|
|
||||||
"X-Requested-With": "XMLHttpRequest",
|
|
||||||
"Sec-Fetch-Dest": "empty",
|
|
||||||
"Sec-Fetch-Mode": "cors",
|
|
||||||
"Sec-Fetch-Site": "same-origin",
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class GestionSportsConnector:
|
class GestionSportsConnector:
|
||||||
|
@ -32,15 +27,15 @@ class GestionSportsConnector:
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def connection_url(self) -> str:
|
def connection_url(self) -> str:
|
||||||
return f"{self.url}/connexion.php?"
|
return urljoin(self.url, "/connexion.php?")
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def login_url(self) -> str:
|
def login_url(self) -> str:
|
||||||
return f"{self.url}/connexion.php?"
|
return urljoin(self.url, "/connexion.php?")
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def booking_url(self) -> str:
|
def booking_url(self) -> str:
|
||||||
return f"{self.url}/membre/reservation.html?"
|
return urljoin(self.url, "/membre/reservation.html?")
|
||||||
|
|
||||||
async def connect(self) -> ClientResponse:
|
async def connect(self) -> ClientResponse:
|
||||||
LOGGER.info("Connecting to GestionSports API")
|
LOGGER.info("Connecting to GestionSports API")
|
||||||
|
@ -48,29 +43,35 @@ class GestionSportsConnector:
|
||||||
await response.text()
|
await response.text()
|
||||||
return response
|
return response
|
||||||
|
|
||||||
async def login(self, user: User) -> ClientResponse:
|
async def login(self, user: User, club: Club) -> ClientResponse:
|
||||||
payload = (
|
payload = (
|
||||||
self.payload_builder.login(user.login)
|
self.payload_builder.login(user.login)
|
||||||
.password(user.password)
|
.password(user.password)
|
||||||
.club_id(user.club_id)
|
.club_id(club.id)
|
||||||
.build_login_payload()
|
.build_login_payload()
|
||||||
)
|
)
|
||||||
|
|
||||||
async with self.session.post(
|
async with self.session.post(
|
||||||
self.login_url, data=payload, headers=HEADERS
|
self.login_url, data=payload, headers=POST_HEADERS
|
||||||
) as response:
|
) as response:
|
||||||
await response.text()
|
await response.text()
|
||||||
return response
|
return response
|
||||||
|
|
||||||
async def book(self, booking_filter: BookingFilter) -> int | None:
|
async def book(self, booking_filter: BookingFilter, club: Club) -> int | None:
|
||||||
|
# use asyncio to request a booking on every court
|
||||||
|
# the gestion-sports backend is able to book only one court for a user
|
||||||
bookings = await asyncio.gather(
|
bookings = await asyncio.gather(
|
||||||
*[
|
*[
|
||||||
self.book_one_court(booking_filter, court_id)
|
self.book_one_court(booking_filter, court_id)
|
||||||
for court_id in booking_filter.court_ids
|
for court_id in club.courts_ids
|
||||||
],
|
],
|
||||||
return_exceptions=True,
|
return_exceptions=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
return await self.get_booked_court(bookings)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
async def get_booked_court(bookings):
|
||||||
for court, is_booked in bookings:
|
for court, is_booked in bookings:
|
||||||
if is_booked:
|
if is_booked:
|
||||||
return court
|
return court
|
||||||
|
@ -79,11 +80,9 @@ class GestionSportsConnector:
|
||||||
async def book_one_court(
|
async def book_one_court(
|
||||||
self, booking_filter: BookingFilter, court_id: int
|
self, booking_filter: BookingFilter, court_id: int
|
||||||
) -> tuple[int, bool]:
|
) -> tuple[int, bool]:
|
||||||
date_format = "%d/%m/%Y"
|
|
||||||
time_format = "%H:%M"
|
|
||||||
payload = (
|
payload = (
|
||||||
self.payload_builder.date(booking_filter.date.date().strftime(date_format))
|
self.payload_builder.date(booking_filter.date.date().strftime(DATE_FORMAT))
|
||||||
.time(booking_filter.date.time().strftime(time_format))
|
.time(booking_filter.date.time().strftime(TIME_FORMAT))
|
||||||
.sport_id(booking_filter.sport_id)
|
.sport_id(booking_filter.sport_id)
|
||||||
.court_id(court_id)
|
.court_id(court_id)
|
||||||
.build_booking_payload()
|
.build_booking_payload()
|
||||||
|
@ -92,7 +91,7 @@ class GestionSportsConnector:
|
||||||
|
|
||||||
async def is_court_booked(self, payload: str) -> bool:
|
async def is_court_booked(self, payload: str) -> bool:
|
||||||
async with self.session.post(
|
async with self.session.post(
|
||||||
self.booking_url, data=payload, headers=HEADERS
|
self.booking_url, data=payload, headers=POST_HEADERS
|
||||||
) as response:
|
) as response:
|
||||||
return self.is_response_status_ok(await response.text())
|
return self.is_response_status_ok(await response.text())
|
||||||
|
|
||||||
|
|
|
@ -1,16 +1,18 @@
|
||||||
from typing import List
|
|
||||||
|
|
||||||
from pydantic import BaseModel, Field
|
from pydantic import BaseModel, Field
|
||||||
from pydantic_extra_types.pendulum_dt import DateTime
|
from pydantic_extra_types.pendulum_dt import DateTime
|
||||||
|
|
||||||
|
|
||||||
|
class Club(BaseModel):
|
||||||
|
id: str = Field()
|
||||||
|
url: str = Field()
|
||||||
|
courts_ids: list[int] = Field(default_factory=list)
|
||||||
|
|
||||||
|
|
||||||
|
class BookingFilter(BaseModel):
|
||||||
|
sport_id: int = Field()
|
||||||
|
date: DateTime = Field()
|
||||||
|
|
||||||
|
|
||||||
class User(BaseModel):
|
class User(BaseModel):
|
||||||
login: str = Field()
|
login: str = Field()
|
||||||
password: str = Field(repr=False)
|
password: str = Field(repr=False)
|
||||||
club_id: str = Field()
|
|
||||||
|
|
||||||
|
|
||||||
class BookingFilter(BaseModel):
|
|
||||||
court_ids: List[int] = Field()
|
|
||||||
sport_id: int = Field()
|
|
||||||
date: DateTime = Field()
|
|
||||||
|
|
12
resa_padel/resources/gestion-sports/post-headers.json
Normal file
12
resa_padel/resources/gestion-sports/post-headers.json
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
{
|
||||||
|
"Connection": "keep-alive",
|
||||||
|
"Accept-Language": "en-US,en;q=0.5",
|
||||||
|
"Accept-Encoding": "gzip, deflate, br",
|
||||||
|
"DNT": "1",
|
||||||
|
"Content-Type": "application/x-www-form-urlencoded; charset=UTF-8",
|
||||||
|
"Accept": "application/json, text/javascript, */*; q=0.01",
|
||||||
|
"X-Requested-With": "XMLHttpRequest",
|
||||||
|
"Sec-Fetch-Dest": "empty",
|
||||||
|
"Sec-Fetch-Mode": "cors",
|
||||||
|
"Sec-Fetch-Site": "same-origin"
|
||||||
|
}
|
|
@ -2,20 +2,20 @@ import json
|
||||||
|
|
||||||
import pendulum
|
import pendulum
|
||||||
import pytest
|
import pytest
|
||||||
from aiohttp import ClientSession
|
|
||||||
|
|
||||||
from resa_padel.gestion_sports.gestion_sports_connector import \
|
from resa_padel.gestion_sports.gestion_sports_payload_builder import (
|
||||||
GestionSportsConnector
|
GestionSportsPayloadBuilder,
|
||||||
from resa_padel.gestion_sports.gestion_sports_payload_builder import \
|
)
|
||||||
GestionSportsPayloadBuilder
|
from resa_padel.models import BookingFilter, Club, User
|
||||||
from resa_padel.models import BookingFilter, User
|
|
||||||
|
|
||||||
user = User(login="padel.testing@jouf.fr", password="ridicule", club_id="123")
|
user = User(login="padel.testing@jouf.fr", password="ridicule", club_id="123")
|
||||||
|
url = "https://tpc.padel.com"
|
||||||
|
club = Club(id="123", url=url, courts_ids=[606, 607, 608])
|
||||||
|
|
||||||
courts = [606, 607, 608]
|
courts = [606, 607, 608]
|
||||||
sport_id = 217
|
sport_id = 217
|
||||||
booking_date = pendulum.now().add(days=6).set(hour=18, minute=0, second=0)
|
booking_date = pendulum.now().add(days=6).set(hour=18, minute=0, second=0)
|
||||||
booking_filter = BookingFilter(court_ids=courts, sport_id=sport_id, date=booking_date)
|
booking_filter = BookingFilter(sport_id=sport_id, date=booking_date)
|
||||||
|
|
||||||
booking_failure_response = json.dumps(
|
booking_failure_response = json.dumps(
|
||||||
{
|
{
|
||||||
|
@ -45,10 +45,6 @@ booking_payload = (
|
||||||
.build_booking_payload()
|
.build_booking_payload()
|
||||||
)
|
)
|
||||||
|
|
||||||
session = ClientSession()
|
|
||||||
gestion_sports_url = "https://toulousepadelclub.gestion-sports.com"
|
|
||||||
gs_connector = GestionSportsConnector(session, gestion_sports_url)
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def a_user() -> User:
|
def a_user() -> User:
|
||||||
|
@ -60,6 +56,11 @@ def a_booking_filter() -> BookingFilter:
|
||||||
return booking_filter
|
return booking_filter
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def a_club() -> Club:
|
||||||
|
return club
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def a_booking_success_response() -> str:
|
def a_booking_success_response() -> str:
|
||||||
return booking_success_response
|
return booking_success_response
|
||||||
|
@ -73,8 +74,3 @@ def a_booking_failure_response() -> str:
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def a_booking_payload() -> str:
|
def a_booking_payload() -> str:
|
||||||
return booking_payload
|
return booking_payload
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
|
||||||
def a_connector():
|
|
||||||
return gs_connector
|
|
||||||
|
|
|
@ -2,60 +2,64 @@ import pytest
|
||||||
from aiohttp import ClientSession
|
from aiohttp import ClientSession
|
||||||
from yarl import URL
|
from yarl import URL
|
||||||
|
|
||||||
from resa_padel.gestion_sports.gestion_sports_connector import \
|
from resa_padel.gestion_sports.gestion_sports_connector import GestionSportsConnector
|
||||||
GestionSportsConnector
|
from tests.fixtures import (
|
||||||
from tests.fixtures import (a_booking_failure_response, a_booking_filter,
|
a_booking_failure_response,
|
||||||
a_booking_payload, a_booking_success_response,
|
a_booking_filter,
|
||||||
a_connector, a_user)
|
a_booking_payload,
|
||||||
|
a_booking_success_response,
|
||||||
|
a_club,
|
||||||
|
a_user,
|
||||||
|
)
|
||||||
|
|
||||||
gestion_sports_url = "https://toulousepadelclub.gestion-sports.com"
|
tpc_url = "https://toulousepadelclub.gestion-sports.com"
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.asyncio
|
@pytest.mark.asyncio
|
||||||
async def test_should_connect_to_gestion_sports_website():
|
async def test_should_connect_to_gestion_sports_website():
|
||||||
async with ClientSession() as session:
|
async with ClientSession() as session:
|
||||||
cookies = session.cookie_jar.filter_cookies(URL(gestion_sports_url))
|
cookies = session.cookie_jar.filter_cookies(URL(tpc_url))
|
||||||
assert cookies.get("PHPSESSID") is None
|
assert cookies.get("PHPSESSID") is None
|
||||||
gs_connector = GestionSportsConnector(session, gestion_sports_url)
|
gs_connector = GestionSportsConnector(session, tpc_url)
|
||||||
|
|
||||||
response = await gs_connector.connect()
|
response = await gs_connector.connect()
|
||||||
|
|
||||||
assert response.status == 200
|
assert response.status == 200
|
||||||
assert response.request_info.method == "GET"
|
assert response.request_info.method == "GET"
|
||||||
assert response.content_type == "text/html"
|
assert response.content_type == "text/html"
|
||||||
assert response.request_info.url == URL(gestion_sports_url + "/connexion.php")
|
assert response.request_info.url == URL(tpc_url + "/connexion.php")
|
||||||
assert response.charset == "UTF-8"
|
assert response.charset == "UTF-8"
|
||||||
|
|
||||||
cookies = session.cookie_jar.filter_cookies(URL(gestion_sports_url))
|
cookies = session.cookie_jar.filter_cookies(URL(tpc_url))
|
||||||
assert cookies.get("PHPSESSID") is not None
|
assert cookies.get("PHPSESSID") is not None
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.asyncio
|
@pytest.mark.asyncio
|
||||||
async def test_should_login_to_gestion_sports_website(a_user):
|
async def test_should_login_to_gestion_sports_website(a_user, a_club):
|
||||||
async with ClientSession() as session:
|
async with ClientSession() as session:
|
||||||
gs_connector = GestionSportsConnector(session, gestion_sports_url)
|
gs_connector = GestionSportsConnector(session, tpc_url)
|
||||||
await gs_connector.connect()
|
await gs_connector.connect()
|
||||||
|
|
||||||
response = await gs_connector.login(a_user)
|
response = await gs_connector.login(a_user, a_club)
|
||||||
|
|
||||||
assert response.status == 200
|
assert response.status == 200
|
||||||
assert response.request_info.url == URL(gestion_sports_url + "/connexion.php")
|
assert response.request_info.url == URL(tpc_url + "/connexion.php")
|
||||||
assert response.request_info.method == "POST"
|
assert response.request_info.method == "POST"
|
||||||
|
|
||||||
cookies = session.cookie_jar.filter_cookies(URL(gestion_sports_url))
|
cookies = session.cookie_jar.filter_cookies(URL(tpc_url))
|
||||||
assert cookies.get("COOK_ID_CLUB").value is not None
|
assert cookies.get("COOK_ID_CLUB").value is not None
|
||||||
assert cookies.get("COOK_ID_USER").value is not None
|
assert cookies.get("COOK_ID_USER").value is not None
|
||||||
assert cookies.get("PHPSESSID") is not None
|
assert cookies.get("PHPSESSID") is not None
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.asyncio
|
@pytest.mark.asyncio
|
||||||
async def test_booking_url_should_be_reachable(a_user, a_booking_filter):
|
async def test_booking_url_should_be_reachable(a_user, a_booking_filter, a_club):
|
||||||
async with ClientSession() as session:
|
async with ClientSession() as session:
|
||||||
gs_connector = GestionSportsConnector(session, gestion_sports_url)
|
gs_connector = GestionSportsConnector(session, tpc_url)
|
||||||
await gs_connector.connect()
|
await gs_connector.connect()
|
||||||
await gs_connector.login(a_user)
|
await gs_connector.login(a_user, a_club)
|
||||||
|
|
||||||
court_booked = await gs_connector.book(a_booking_filter)
|
court_booked = await gs_connector.book(a_booking_filter, a_club)
|
||||||
# At 18:00 no chance to get a booking, any day of the week
|
# At 18:00 no chance to get a booking, any day of the week
|
||||||
assert court_booked is None
|
assert court_booked is None
|
||||||
|
|
||||||
|
@ -64,24 +68,25 @@ async def test_booking_url_should_be_reachable(a_user, a_booking_filter):
|
||||||
async def test_should_book_a_court_from_gestion_sports(
|
async def test_should_book_a_court_from_gestion_sports(
|
||||||
aioresponses,
|
aioresponses,
|
||||||
a_booking_filter,
|
a_booking_filter,
|
||||||
|
a_club,
|
||||||
a_booking_success_response,
|
a_booking_success_response,
|
||||||
a_booking_failure_response,
|
a_booking_failure_response,
|
||||||
):
|
):
|
||||||
booking_url = URL(gestion_sports_url + "/membre/reservation.html?")
|
booking_url = URL(tpc_url + "/membre/reservation.html?")
|
||||||
|
|
||||||
# first booking request will fail
|
# first booking request will fail
|
||||||
aioresponses.post(URL(booking_url), status=200, body=a_booking_failure_response)
|
aioresponses.post(booking_url, status=200, body=a_booking_failure_response)
|
||||||
# first booking request will succeed
|
# first booking request will succeed
|
||||||
aioresponses.post(URL(booking_url), status=200, body=a_booking_success_response)
|
aioresponses.post(booking_url, status=200, body=a_booking_success_response)
|
||||||
# first booking request will fail
|
# first booking request will fail
|
||||||
aioresponses.post(URL(booking_url), status=200, body=a_booking_failure_response)
|
aioresponses.post(booking_url, status=200, body=a_booking_failure_response)
|
||||||
|
|
||||||
async with ClientSession() as session:
|
async with ClientSession() as session:
|
||||||
gs_connector = GestionSportsConnector(session, gestion_sports_url)
|
gs_connector = GestionSportsConnector(session, tpc_url)
|
||||||
court_booked = await gs_connector.book(a_booking_filter)
|
court_booked = await gs_connector.book(a_booking_filter, a_club)
|
||||||
|
|
||||||
# the second element of the list is the booked court
|
# the second element of the list is the booked court
|
||||||
assert court_booked == a_booking_filter.court_ids[1]
|
assert court_booked == a_club.courts_ids[1]
|
||||||
|
|
||||||
|
|
||||||
def test_response_status_should_be_ok(a_booking_success_response):
|
def test_response_status_should_be_ok(a_booking_success_response):
|
||||||
|
@ -89,28 +94,33 @@ def test_response_status_should_be_ok(a_booking_success_response):
|
||||||
assert is_booked
|
assert is_booked
|
||||||
|
|
||||||
|
|
||||||
def test_response_status_should_be_not_ok(aioresponses, a_booking_failure_response):
|
def test_response_status_should_be_not_ok(a_booking_failure_response):
|
||||||
is_booked = GestionSportsConnector.is_response_status_ok(a_booking_failure_response)
|
is_booked = GestionSportsConnector.is_response_status_ok(a_booking_failure_response)
|
||||||
assert not is_booked
|
assert not is_booked
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.asyncio
|
@pytest.mark.asyncio
|
||||||
async def test_court_should_not_be_booked(
|
async def test_court_should_not_be_booked(
|
||||||
aioresponses, a_connector, a_booking_payload, a_booking_failure_response
|
aioresponses, a_booking_payload, a_booking_failure_response
|
||||||
):
|
):
|
||||||
aioresponses.post(
|
async with ClientSession() as session:
|
||||||
URL(a_connector.booking_url), status=200, body=a_booking_failure_response
|
tpc_connector = GestionSportsConnector(session, tpc_url)
|
||||||
)
|
aioresponses.post(
|
||||||
is_booked = await a_connector.is_court_booked(a_booking_payload)
|
URL(tpc_connector.booking_url), status=200, body=a_booking_failure_response
|
||||||
assert not is_booked
|
)
|
||||||
|
is_booked = await tpc_connector.is_court_booked(a_booking_payload)
|
||||||
|
assert not is_booked
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.asyncio
|
@pytest.mark.asyncio
|
||||||
async def test_court_should_be_booked(
|
async def test_court_should_be_booked(
|
||||||
aioresponses, a_connector, a_booking_payload, a_booking_success_response
|
aioresponses, a_booking_payload, a_booking_success_response
|
||||||
):
|
):
|
||||||
aioresponses.post(
|
async with ClientSession() as session:
|
||||||
URL(a_connector.booking_url), status=200, body=a_booking_success_response
|
tpc_connector = GestionSportsConnector(session, tpc_url)
|
||||||
)
|
|
||||||
is_booked = await a_connector.is_court_booked(a_booking_payload)
|
aioresponses.post(
|
||||||
assert is_booked
|
URL(tpc_connector.booking_url), status=200, body=a_booking_success_response
|
||||||
|
)
|
||||||
|
is_booked = await tpc_connector.is_court_booked(a_booking_payload)
|
||||||
|
assert is_booked
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
import pendulum
|
import pendulum
|
||||||
|
|
||||||
from resa_padel.gestion_sports.gestion_sports_payload_builder import \
|
from resa_padel.gestion_sports.gestion_sports_payload_builder import (
|
||||||
GestionSportsPayloadBuilder
|
GestionSportsPayloadBuilder,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def test_login_payload_should_be_built():
|
def test_login_payload_should_be_built():
|
||||||
|
|
|
@ -1,66 +1,109 @@
|
||||||
import asyncio
|
import asyncio
|
||||||
|
import os
|
||||||
|
from unittest.mock import patch
|
||||||
|
from urllib.parse import urljoin
|
||||||
|
|
||||||
import pendulum
|
|
||||||
from aioresponses import aioresponses
|
from aioresponses import aioresponses
|
||||||
|
|
||||||
from resa_padel import booking
|
from resa_padel import booking
|
||||||
from resa_padel.models import BookingFilter, User
|
from tests import fixtures
|
||||||
from tests.fixtures import (a_booking_failure_response,
|
from tests.fixtures import (
|
||||||
a_booking_success_response)
|
a_booking_failure_response,
|
||||||
|
a_booking_filter,
|
||||||
|
a_booking_success_response,
|
||||||
|
a_club,
|
||||||
|
a_user,
|
||||||
|
)
|
||||||
|
|
||||||
login = "user"
|
login = "user"
|
||||||
password = "password"
|
password = "password"
|
||||||
club_id = "98"
|
club_id = "88"
|
||||||
court_id = "11"
|
court_id = "11"
|
||||||
|
|
||||||
|
|
||||||
# FIXME
|
def mock_successful_connection(aio_mock, url):
|
||||||
# check that called are passed to the given urls
|
aio_mock.get(
|
||||||
# check made with cookies, but at the current time, cookies from the response are
|
url,
|
||||||
# not set in the session. Why ? I don't know....
|
status=200,
|
||||||
|
headers={"Set-Cookie": f"connection_called=True; Domain={url}"},
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def mock_successful_login(aio_mock, url):
|
||||||
|
aio_mock.post(
|
||||||
|
url,
|
||||||
|
status=200,
|
||||||
|
headers={"Set-Cookie": f"login_called=True; Domain={url}"},
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def mock_booking(aio_mock, url, response):
|
||||||
|
aio_mock.post(
|
||||||
|
url,
|
||||||
|
status=200,
|
||||||
|
headers={"Set-Cookie": f"booking_called=True; Domain={url}"},
|
||||||
|
body=response,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def mock_rest_api_from_connection_to_booking(
|
||||||
|
aio_mock, url, a_booking_failure_response, a_booking_success_response
|
||||||
|
):
|
||||||
|
connexion_url = urljoin(url, "/connexion.php?")
|
||||||
|
mock_successful_connection(aio_mock, connexion_url)
|
||||||
|
login_url = urljoin(url, "/connexion.php?")
|
||||||
|
mock_successful_login(aio_mock, login_url)
|
||||||
|
booking_url = urljoin(url, "/membre/reservation.html?")
|
||||||
|
mock_booking(aio_mock, booking_url, a_booking_failure_response)
|
||||||
|
mock_booking(aio_mock, booking_url, a_booking_success_response)
|
||||||
|
|
||||||
|
|
||||||
def test_booking_does_the_rights_calls(
|
def test_booking_does_the_rights_calls(
|
||||||
a_booking_success_response, a_booking_failure_response
|
a_booking_success_response,
|
||||||
|
a_booking_failure_response,
|
||||||
|
a_user,
|
||||||
|
a_club,
|
||||||
|
a_booking_filter,
|
||||||
):
|
):
|
||||||
# mock connection to the booking platform
|
# mock connection to the booking platform
|
||||||
platform_url = "https://some.url"
|
with aioresponses() as aio_mock:
|
||||||
connection_url = platform_url + "/connexion.php"
|
mock_rest_api_from_connection_to_booking(
|
||||||
login_url = connection_url
|
aio_mock,
|
||||||
booking_url = platform_url + "/membre/reservation.html"
|
fixtures.url,
|
||||||
|
a_booking_failure_response,
|
||||||
|
a_booking_success_response,
|
||||||
|
)
|
||||||
|
|
||||||
loop = asyncio.get_event_loop()
|
loop = asyncio.get_event_loop()
|
||||||
|
|
||||||
|
court_booked = loop.run_until_complete(
|
||||||
|
booking.book(a_club, a_user, a_booking_filter)
|
||||||
|
)
|
||||||
|
assert court_booked == a_club.courts_ids[1]
|
||||||
|
|
||||||
|
|
||||||
|
@patch.dict(
|
||||||
|
os.environ,
|
||||||
|
{
|
||||||
|
"LOGIN": login,
|
||||||
|
"PASSWORD": password,
|
||||||
|
"CLUB_ID": club_id,
|
||||||
|
"CLUB_URL": fixtures.url,
|
||||||
|
"COURT_IDS": "7,8,10",
|
||||||
|
"SPORT_ID": "217",
|
||||||
|
"DATE_TIME": "2024-04-23T15:00:00Z",
|
||||||
|
},
|
||||||
|
clear=True,
|
||||||
|
)
|
||||||
|
def test_main(a_booking_success_response, a_booking_failure_response):
|
||||||
|
|
||||||
with aioresponses() as aio_mock:
|
with aioresponses() as aio_mock:
|
||||||
aio_mock.get(
|
mock_rest_api_from_connection_to_booking(
|
||||||
connection_url,
|
aio_mock,
|
||||||
status=200,
|
fixtures.url,
|
||||||
headers={"Set-Cookie": f"connection_called=True; Domain={platform_url}"},
|
a_booking_failure_response,
|
||||||
)
|
a_booking_success_response,
|
||||||
aio_mock.post(
|
|
||||||
login_url,
|
|
||||||
status=200,
|
|
||||||
headers={"Set-Cookie": f"login_called=True; Domain={platform_url}"},
|
|
||||||
)
|
|
||||||
aio_mock.post(
|
|
||||||
booking_url,
|
|
||||||
status=200,
|
|
||||||
headers={"Set-Cookie": f"booking_called=True; Domain={platform_url}"},
|
|
||||||
body=a_booking_failure_response,
|
|
||||||
)
|
|
||||||
aio_mock.post(
|
|
||||||
booking_url,
|
|
||||||
status=200,
|
|
||||||
headers={"Set-Cookie": f"booking_called=True; Domain={platform_url}"},
|
|
||||||
body=a_booking_success_response,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
user = User(login=login, password=password, club_id=club_id)
|
court_booked = booking.main()
|
||||||
booking_filter = BookingFilter(
|
assert court_booked == 8
|
||||||
court_ids=[607, 606], sport_id=444, date=pendulum.now().add(days=6)
|
|
||||||
)
|
|
||||||
|
|
||||||
loop.run_until_complete(booking.book(platform_url, user, booking_filter))
|
|
||||||
|
|
||||||
# cookies = session.cookie_jar.filter_cookies(platform_url)
|
|
||||||
# assert cookies.get("connection_called") == "True"
|
|
||||||
# assert cookies.get("login_called") == "True"
|
|
||||||
# assert cookies.get("booking_called") == "True"
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue