diff --git a/source/jormungandr/jormungandr/instance.py b/source/jormungandr/jormungandr/instance.py index dd862e8b55..b19724b1f5 100644 --- a/source/jormungandr/jormungandr/instance.py +++ b/source/jormungandr/jormungandr/instance.py @@ -63,6 +63,7 @@ from navitiacommon import default_values from jormungandr.equipments import EquipmentProviderManager from jormungandr.external_services import ExternalServiceManager +from jormungandr.parking_space_availability.bss.bss_provider_manager import BssProviderManager from jormungandr.utils import ( can_connect_to_database, make_origin_destination_key, @@ -169,6 +170,7 @@ def __init__( use_multi_reverse=False, resp_content_limit_bytes=None, resp_content_limit_endpoints_whitelist=None, + individual_bss_provider=[], ): super(Instance, self).__init__( name=name, @@ -257,6 +259,15 @@ def __init__( self.external_service_provider_manager = ExternalServiceManager( self, external_service_provider_configurations, self.get_external_service_providers_from_db ) + + # Init BSS provider manager from config from external services in bdd + if disable_database: + self.bss_provider_manager = BssProviderManager(individual_bss_provider) + else: + self.bss_provider_manager = BssProviderManager( + individual_bss_provider, self.get_bss_stations_services_from_db + ) + self.external_service_provider_manager.init_external_services() self.instance_db = instance_db self._ghost_words = ghost_words or [] @@ -333,6 +344,14 @@ def get_realtime_proxies_from_db(self): result = models.external_services if models else None return [res for res in result if res.navitia_service == 'realtime_proxies'] + def get_bss_stations_services_from_db(self): + """ + :return: a callable query of external services associated to the current instance in db + """ + models = self._get_models() + result = models.external_services if models else [] + return [res for res in result if res.navitia_service == 'bss_stations'] + @property def autocomplete(self): if self._autocomplete_type: @@ -987,6 +1006,9 @@ def get_all_street_networks(self): def get_all_ridesharing_services(self): return self.ridesharing_services_manager.get_all_ridesharing_services() + def get_all_bss_providers(self): + return self.bss_provider_manager.get_providers() + def get_autocomplete(self, requested_autocomplete): if not requested_autocomplete: return self.autocomplete diff --git a/source/jormungandr/jormungandr/instance_manager.py b/source/jormungandr/jormungandr/instance_manager.py index 80ac5fe246..1e94e2e079 100644 --- a/source/jormungandr/jormungandr/instance_manager.py +++ b/source/jormungandr/jormungandr/instance_manager.py @@ -128,6 +128,7 @@ def register_instance(self, config): use_multi_reverse=config.get('use_multi_reverse', False), resp_content_limit_bytes=config.get('resp_content_limit_bytes', None), resp_content_limit_endpoints_whitelist=config.get('resp_content_limit_endpoints_whitelist', None), + individual_bss_provider=config.get('individual_bss_provider', []), ) self.instances[instance.name] = instance diff --git a/source/jormungandr/jormungandr/interfaces/v1/__init__.py b/source/jormungandr/jormungandr/interfaces/v1/__init__.py index 0265c65681..b6968c8ded 100644 --- a/source/jormungandr/jormungandr/interfaces/v1/__init__.py +++ b/source/jormungandr/jormungandr/interfaces/v1/__init__.py @@ -56,6 +56,10 @@ def add_common_status(response, instance): for rss in instance.get_all_ridesharing_services(): response['status']['ridesharing_services'].append(rss.status()) + response['status']['bss_providers'] = [] + for bp in instance.get_all_bss_providers(): + response['status']['bss_providers'].append(bp.status()) + response['status']['equipment_providers_services'] = {} response['status']['equipment_providers_services'][ 'equipment_providers_keys' diff --git a/source/jormungandr/jormungandr/interfaces/v1/serializer/status.py b/source/jormungandr/jormungandr/interfaces/v1/serializer/status.py index eabba34b25..ca9745e0b3 100644 --- a/source/jormungandr/jormungandr/interfaces/v1/serializer/status.py +++ b/source/jormungandr/jormungandr/interfaces/v1/serializer/status.py @@ -204,6 +204,12 @@ def get_arrival_radius(self, obj): return obj.get('arrival_radius') +class BSSStationsServiceSerializer(OutsideServiceCommon): + id = Field(display_none=True) + url = Field(display_none=True) + class_ = Field(schema_type=str, label='class', attr='class') + + class EquipmentProvidersSerializer(NullableDictSerializer): key = Field(schema_type=str, display_none=False) codes_types = Field(schema_type=str, many=True, display_none=True) @@ -255,6 +261,7 @@ class CommonStatusSerializer(NullableDictSerializer): publication_date = Field(schema_type=str, display_none=False) street_networks = StreetNetworkSerializer(many=True, display_none=False) ridesharing_services = RidesharingServicesSerializer(many=True, display_none=False) + bss_providers = BSSStationsServiceSerializer(many=True, display_none=False) equipment_providers_services = EquipmentProvidersServicesSerializer(display_none=False) external_providers_services = ExternalServiceProvidersServicesSerializer(display_none=False) start_production_date = Field(schema_type=str, display_none=False) diff --git a/source/jormungandr/jormungandr/parking_space_availability/bss/bss_provider_manager.py b/source/jormungandr/jormungandr/parking_space_availability/bss/bss_provider_manager.py index 021fc11992..014292ba90 100644 --- a/source/jormungandr/jormungandr/parking_space_availability/bss/bss_provider_manager.py +++ b/source/jormungandr/jormungandr/parking_space_availability/bss/bss_provider_manager.py @@ -67,6 +67,7 @@ def update_config(self): self._last_update = datetime.datetime.utcnow() try: + # BSS provider list from the database (table bss_provider) providers = self._providers_getter() except Exception as e: logger.exception('No access to table bss_provider (error: {})'.format(e)) @@ -111,11 +112,13 @@ def _handle_poi(self, item): return provider return None - # TODO use public version everywhere def _get_providers(self): self.update_config() - # providers from the database have priority on legacies providers return list(self._bss_providers.values()) + self._bss_providers_legacy def get_providers(self): return self._get_providers() + + def exist_provider(self): + self.update_config() + return any(self.get_providers()) diff --git a/source/jormungandr/jormungandr/parking_space_availability/bss/forseti.py b/source/jormungandr/jormungandr/parking_space_availability/bss/forseti.py new file mode 100644 index 0000000000..52dff87235 --- /dev/null +++ b/source/jormungandr/jormungandr/parking_space_availability/bss/forseti.py @@ -0,0 +1,146 @@ +# coding: utf-8 + +# Copyright (c) 2001-2022, Hove and/or its affiliates. All rights reserved. +# +# This file is part of Navitia, +# the software to build cool stuff with public transport. +# +# Hope you'll enjoy and contribute to this project, +# powered by Hove (www.hove.com). +# Help us simplify mobility and open public transport: +# a non ending quest to the responsive locomotion way of traveling! +# +# LICENCE: This program is free software; you can redistribute it and/or modify +# it under the terms of the GNU Affero General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU Affero General Public License for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +# Stay tuned using +# twitter @navitia +# channel `#navitia` on riot https://riot.im/app/#/room/#navitia:matrix.org +# https://groups.google.com/d/forum/navitia +# www.navitia.io +from __future__ import absolute_import, print_function, unicode_literals, division +from jormungandr import cache, app +import pybreaker +import logging +import requests as requests +from jormungandr.ptref import FeedPublisher +from jormungandr.parking_space_availability.bss.stands import Stands, StandsStatus +from jormungandr.parking_space_availability.bss.common_bss_provider import CommonBssProvider, BssProxyError +import six + +DEFAULT_FORSETI_FEED_PUBLISHER = { + 'id': 'forseti', + 'name': 'forseti', + 'license': 'Private', + 'url': 'www.navitia.io', +} + + +class ForsetiProvider(CommonBssProvider): + """ + class managing calls to Forseti external service providing real-time BSS stands availability + + """ + + def __init__( + self, + service_url, + distance=50, + organizations=[], + feed_publisher=DEFAULT_FORSETI_FEED_PUBLISHER, + timeout=2, + **kwargs + ): + self.logger = logging.getLogger(__name__) + self.service_url = service_url + self.distance = distance + self.timeout = timeout + self.network = "Forseti" + self.breaker = pybreaker.CircuitBreaker( + fail_max=kwargs.get('circuit_breaker_max_fail', app.config['CIRCUIT_BREAKER_MAX_FORSETI_FAIL']), + reset_timeout=kwargs.get( + 'circuit_breaker_reset_timeout', app.config['CIRCUIT_BREAKER_FORSETI_TIMEOUT_S'] + ), + ) + + self._feed_publisher = FeedPublisher(**feed_publisher) if feed_publisher else None + if not isinstance(organizations, list): + import json + self.organizations = json.loads(str(organizations)) + else: + self.organizations = organizations + + def service_caller(self, method, url): + try: + response = self.breaker.call(method, url, timeout=self.timeout, verify=False) + if not response or response.status_code != 200: + logging.getLogger(__name__).error( + 'Forseti, Invalid response, status_code: {}'.format(response.status_code) + ) + raise BssProxyError('non 200 response') + return response + except pybreaker.CircuitBreakerError as e: + logging.getLogger(__name__).error('forseti service dead (error: {})'.format(e)) + raise BssProxyError('circuit breaker open') + except requests.Timeout as t: + logging.getLogger(__name__).error('forseti service timeout (error: {})'.format(t)) + raise BssProxyError('timeout') + except Exception as e: + logging.getLogger(__name__).exception('forseti error : {}'.format(str(e))) + raise BssProxyError(str(e)) + + @cache.memoize(app.config.get(str('CACHE_CONFIGURATION'), {}).get(str('TIMEOUT_FORSETI'), 30)) + def _call_webservice(self, arguments): + url = "{}?{}".format(self.service_url, arguments) + data = self.service_caller(method=requests.get, url=url) + return data.json() + + def support_poi(self, poi): + return True + + def status(self): + # return {'network': self.network, 'operators': self.operators} + return { + 'id': six.text_type(self.network), + 'url': self.service_url, + 'class': self.__class__.__name__, + } + + def feed_publisher(self): + return self._feed_publisher + + def _get_informations(self, poi): + longitude = poi.get('coord', {}).get('lon', None) + latitude = poi.get('coord', {}).get('lat', None) + if latitude is None or longitude is None: + return Stands(0, 0, StandsStatus.unavailable) + + params_organizations = '' + for param in self.organizations: + params_organizations += '&organization[]={}'.format(param) + + # /stations?coord=lon%3Blat&distance=self.distance&organization[]=org1&organization[]=org2 ... + arguments = 'coord={}%3B{}&distance={}{}'.format( + longitude, latitude, self.distance, params_organizations + ) + data = self._call_webservice(arguments) + + if not data: + return Stands(0, 0, StandsStatus.unavailable) + obj_stations = data.get('stations', []) + + if not obj_stations: + return Stands(0, 0, StandsStatus.unavailable) + + vehicle_count = sum((v.get('count', 0) for v in obj_stations[0].get('vehicles', {}))) + return Stands(obj_stations[0].get('docks', {}).get('available', 0), vehicle_count, StandsStatus.open) diff --git a/source/jormungandr/jormungandr/parking_space_availability/bss/tests/forseti_test.py b/source/jormungandr/jormungandr/parking_space_availability/bss/tests/forseti_test.py new file mode 100644 index 0000000000..21b65efab9 --- /dev/null +++ b/source/jormungandr/jormungandr/parking_space_availability/bss/tests/forseti_test.py @@ -0,0 +1,75 @@ +# encoding: utf-8 +# Copyright (c) 2001-2022, Hove and/or its affiliates. All rights reserved. +# +# This file is part of Navitia, +# the software to build cool stuff with public transport. +# +# Hope you'll enjoy and contribute to this project, +# powered by Hove (www.hove.com). +# Help us simplify mobility and open public transport: +# a non ending quest to the responsive locomotion way of traveling! +# +# LICENCE: This program is free software; you can redistribute it and/or modify +# it under the terms of the GNU Affero General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU Affero General Public License for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +# Stay tuned using +# twitter @navitia +# channel `#navitia` on riot https://riot.im/app/#/room/#navitia:matrix.org +# https://groups.google.com/d/forum/navitia +# www.navitia.io + +from __future__ import absolute_import, print_function, unicode_literals, division +from copy import deepcopy +from jormungandr.parking_space_availability.bss.forseti import ForsetiProvider +from jormungandr.parking_space_availability.bss.stands import Stands, StandsStatus +from mock import MagicMock + +poi = { + 'poi_type': {'name': 'station vls', 'id': 'poi_type:amenity:bicycle_rental'}, + 'coord': {'lat': '48.0981147', 'lon': '-1.6552921'}, +} + + +def parking_space_availability_forseti_support_poi_test(): + """ + ForsetiProvider bss provider support + Since we search bss station in forseti with coordinate, it is always True + """ + provider = ForsetiProvider('http://forseti') + poi_copy = deepcopy(poi) + assert provider.support_poi(poi_copy) + + +def parking_space_availability_forseti_get_informations_test(): + webservice_response = { + "stations": [ + { + "id": "TAN:Station:18", + "name": "018-VIARME", + "coord": {"lat": 48.0981147, "lon": -1.6552921}, + "vehicles": [{"type": "bicycle", "count": 9}], + "docks": {"available": 4, "total": 13}, + "status": "OPEN", + } + ], + "pagination": {"start_page": 0, "items_on_page": 2, "items_per_page": 25, "total_result": 2}, + } + + provider = ForsetiProvider('http://forseti') + provider._call_webservice = MagicMock(return_value=webservice_response) + assert provider.get_informations(poi) == Stands(4, 9, StandsStatus.open) + + provider._call_webservice = MagicMock(return_value=None) + assert provider.get_informations(poi) == Stands(0, 0, StandsStatus.unavailable) + invalid_poi = {} + assert provider.get_informations(invalid_poi) == Stands(0, 0, StandsStatus.unavailable) diff --git a/source/jormungandr/jormungandr/parking_space_availability/parking_places_manager.py b/source/jormungandr/jormungandr/parking_space_availability/parking_places_manager.py index 31ef2858db..578ac6fe13 100644 --- a/source/jormungandr/jormungandr/parking_space_availability/parking_places_manager.py +++ b/source/jormungandr/jormungandr/parking_space_availability/parking_places_manager.py @@ -80,7 +80,21 @@ def wrapper(*args, **kwargs): bss_provider_manager, self.attribute, self.logger, - 'Error while handling BSS realtime availability', + 'Error while handling global BSS realtime availability', + ) + + if ( + show_bss_stands + and instance + and instance.bss_provider_manager + and instance.bss_provider_manager.exist_provider() + ): + _handle( + response, + instance.bss_provider_manager, + self.attribute, + self.logger, + 'Error while handling individual BSS realtime availability', ) if show_car_park and instance and instance.car_park_provider: @@ -89,7 +103,7 @@ def wrapper(*args, **kwargs): car_park_provider_manager, self.attribute, self.logger, - 'Error while handling car park realtime availability', + f'Error while handling global car park realtime availability with configuration for instance: {instance}', ) return response, status, h diff --git a/source/navitiacommon/navitiacommon/constants.py b/source/navitiacommon/navitiacommon/constants.py index 0630c25ec2..b4cd6c20be 100644 --- a/source/navitiacommon/navitiacommon/constants.py +++ b/source/navitiacommon/navitiacommon/constants.py @@ -34,4 +34,10 @@ ENUM_SHAPE_SCOPE = ('admin', 'street', 'addr', 'poi', 'stop') DEFAULT_SHAPE_SCOPE = ('admin', 'street', 'addr', 'poi') -ENUM_EXTERNAL_SERVICE = ('free_floatings', 'vehicle_occupancies', 'realtime_proxies', 'vehicle_positions') +ENUM_EXTERNAL_SERVICE = ( + 'free_floatings', + 'vehicle_occupancies', + 'realtime_proxies', + 'vehicle_positions', + 'bss_stations', +) diff --git a/source/navitiacommon/navitiacommon/models/external_service.py b/source/navitiacommon/navitiacommon/models/external_service.py index a2d6055658..5fb9c2eab4 100644 --- a/source/navitiacommon/navitiacommon/models/external_service.py +++ b/source/navitiacommon/navitiacommon/models/external_service.py @@ -77,3 +77,9 @@ def get_default(cls, navitia_service=None): def last_update(self): return self.updated_at if self.updated_at else self.created_at + + def full_args(self): + """ + generate args form jormungandr implementation of a bss providers from configuration in external service + """ + return self.args diff --git a/source/tyr/migrations/versions/cab8323f71bd_add_bss_stations_in_navitia_service_type.py b/source/tyr/migrations/versions/cab8323f71bd_add_bss_stations_in_navitia_service_type.py new file mode 100644 index 0000000000..85599b4175 --- /dev/null +++ b/source/tyr/migrations/versions/cab8323f71bd_add_bss_stations_in_navitia_service_type.py @@ -0,0 +1,33 @@ +"""Add bss_stations in navitia_service_type + +Revision ID: cab8323f71bd +Revises: 20d8caa23b26 +Create Date: 2023-09-06 14:19:03.579738 + +""" + +# revision identifiers, used by Alembic. +revision = 'cab8323f71bd' +down_revision = '20d8caa23b26' + +from alembic import op + + +def upgrade(): + op.execute("COMMIT") # See https://bitbucket.org/zzzeek/alembic/issue/123 + op.execute("ALTER TYPE navitia_service_type ADD VALUE 'bss_stations'") + + +def downgrade(): + op.execute( + "DELETE FROM associate_instance_external_service WHERE external_service_id in (SELECT ID FROM external_service WHERE navitia_service = 'bss_stations')" + ) + op.execute("DELETE FROM external_service WHERE navitia_service = 'bss_stations'") + op.execute("ALTER TABLE external_service ALTER COLUMN navitia_service TYPE text") + op.execute("DROP TYPE navitia_service_type CASCADE") + op.execute( + "CREATE TYPE navitia_service_type AS ENUM ('free_floatings', 'vehicle_occupancies', 'realtime_proxies', 'vehicle_positions')" + ) + op.execute( + "ALTER TABLE external_service ALTER COLUMN navitia_service TYPE navitia_service_type USING navitia_service::navitia_service_type" + )