From e1981063359b0910614e5ea3f861a5ce1e54970d Mon Sep 17 00:00:00 2001 From: Christian Struck Date: Sat, 2 Feb 2013 04:17:03 +0100 Subject: Added a new object to return which is easier to handle an implements a merge function if it is wished to mix actualdeparture with scheduleddeparture departures. --- bvggrabber/api/__init__.py | 31 +++++++++++++++++++++++++++++++ bvggrabber/api/actualdeparture.py | 17 +++++++++++------ bvggrabber/api/scheduleddeparture.py | 21 +++++++++++++-------- tests/test_api.py | 32 +++++++++++++++++++++++++++++++- 4 files changed, 86 insertions(+), 15 deletions(-) diff --git a/bvggrabber/api/__init__.py b/bvggrabber/api/__init__.py index 36c913b..c489e8b 100644 --- a/bvggrabber/api/__init__.py +++ b/bvggrabber/api/__init__.py @@ -30,6 +30,37 @@ class QueryApi(object): "the call() method!") +class Response(object): + + def __init__(self, state, departures, error=None): + self._state = state + self._departures = departures + self._error = error + + def merge(self, other): + if isinstance(other, Response): + if not other.state: + raise ValueError("The response contains errors: " + str(other.error)) + elif not self.state: + raise ValueError("The response contains errors: " + str(self.error)) + else: + self.departures.extend(other.departures) + else: + raise TypeError("The given object is not a response object") + + @property + def state(self): + return self._state + + @property + def departures(self): + return self._departures + + @property + def error(self): + return self._error + + @total_ordering class Departure(object): diff --git a/bvggrabber/api/actualdeparture.py b/bvggrabber/api/actualdeparture.py index 93213ef..7f091cc 100644 --- a/bvggrabber/api/actualdeparture.py +++ b/bvggrabber/api/actualdeparture.py @@ -24,7 +24,7 @@ class ActualDepartureQueryApi(QueryApi): def call(self): params = {'input': self.station_enc} response = requests.get(ACTUAL_API_ENDPOINT, params=params) - if response.status_code == requests.codes.ok: + if response.ok: soup = BeautifulSoup(response.text) if soup.find_all('form'): # The station we are looking for is ambiguous or does not exist @@ -32,16 +32,16 @@ class ActualDepartureQueryApi(QueryApi): if stations: # The station is ambiguous stationlist = [s.get('value') for s in stations] - return (False, stationlist) + return Response(False, stationlist) else: # The station does not exist - return (False, []) + return Response(False, []) else: # The station seems to exist result = soup.find('div', {'id': '', 'class': 'ivu_result_box'}) if result is None: - return (False, []) + return Response(False, []) rows = result.find_all('tr') departures = [] for row in rows: @@ -52,6 +52,11 @@ class ActualDepartureQueryApi(QueryApi): when=td[0].text.strip(), line=td[1].text.strip()) departures.append(dep) - return (True, departures) + return Response(True, departures) else: - response.raise_for_status() + try: + response.raise_for_status() + except RequestException as e: + return Response(False, [], e) + else: + return Response(False, [], Exception("An unknown error occured")) diff --git a/bvggrabber/api/scheduleddeparture.py b/bvggrabber/api/scheduleddeparture.py index ba8a453..e144d58 100644 --- a/bvggrabber/api/scheduleddeparture.py +++ b/bvggrabber/api/scheduleddeparture.py @@ -5,7 +5,7 @@ import datetime from bs4 import BeautifulSoup -from bvggrabber.api import QueryApi, Departure +from bvggrabber.api import QueryApi, Departure, Response from bvggrabber.utils.format import dateformat, int2bin, timeformat @@ -37,7 +37,7 @@ class ScheduledDepartureQueryApi(QueryApi): else: raise ValueError("Invalid type for station") self.station = station - self.vehicles = int2bin(vehicles) + self.vehicles = int2bin(vehicles, 7) self.limit = limit def call(self): @@ -49,7 +49,7 @@ class ScheduledDepartureQueryApi(QueryApi): 'maxJourneys': self.limit, 'start': 'yes'} response = requests.get(SCHEDULED_API_ENDPOINT, params=params) - if response.status_code == requests.codes.ok: + if response.ok: soup = BeautifulSoup(response.text) if soup.find('span', 'error'): # The station we are looking for is ambiguous or does not exist @@ -57,15 +57,15 @@ class ScheduledDepartureQueryApi(QueryApi): if stations: # The station is ambiguous stationlist = [s.text.strip() for s in stations] - return (False, stationlist) + return Response(False, stationlist) else: # The station does not exist - return (False, []) + return Response(False, []) else: # The station seems to exist tbody = soup.find('tbody') if tbody is None: - return (False, []) + return Response(False, []) rows = tbody.find_all('tr') departures = [] for row in rows: @@ -75,6 +75,11 @@ class ScheduledDepartureQueryApi(QueryApi): when=tds[0].text.strip(), line=tds[1].text.strip()) departures.append(dep) - return (True, departures) + return Response(True, departures) else: - response.raise_for_status() + try: + response.raise_for_status() + except RequestException as e: + return Response(False, [], e) + else: + return Response(False, [], Exception("An unknown error occured")) diff --git a/tests/test_api.py b/tests/test_api.py index cf111d7..113bbbf 100644 --- a/tests/test_api.py +++ b/tests/test_api.py @@ -3,7 +3,7 @@ import datetime import json import unittest -from bvggrabber.api import QueryApi, Departure, compute_remaining +from bvggrabber.api import QueryApi, Departure, Response, compute_remaining from bvggrabber.utils.format import fullformat @@ -42,6 +42,36 @@ class TestFunctions(unittest.TestCase): 1357092240, 1357092241) +class TestResponse(unittest.TestCase): + + def test_merge(self): + departures = [ Departure("ERP", + "HBF", + datetime.datetime(2013, 1, 2, 3, 4, 1), + "U2"), + Departure("HBF", + "ERP", + datetime.datetime(2013, 2, 1, 3, 4, 1), + "U55")] + departures2 = [ Departure("ERP", + "HBF", + datetime.datetime(2013, 1, 2, 3, 4, 1), + "U6"), + Departure("HBF", + "ERP", + datetime.datetime(2013, 2, 1, 3, 4, 1), + "U9")] + allDepartures = departures + departures2 + r1 = Response(True, departures) + r2 = Response(True, departures2) + r3 = Response(False, []) + self.assertRaises(ValueError, r1.merge, r3) + self.assertRaises(ValueError, r3.merge, r2) + self.assertRaises(TypeError, r1.merge, departures) + r1.merge(r2) + self.assertEqual(r1.departures, allDepartures) + + class TestQueryApi(unittest.TestCase): def test_call(self): -- cgit v1.2.3