Compare commits

..

No commits in common. "80a788933d906684b55710553c5fbc1c6fdd8bb8" and "e09e8f29fb71bd3499485cc0f65404385434e14d" have entirely different histories.

11 changed files with 288 additions and 272 deletions

View File

@ -2,27 +2,25 @@
Small module that makes your journey with RMV REST API somehow easier. Based fully on official RMV API reference and HAFAS documentation. Small module that makes your journey with RMV REST API somehow easier. Based fully on official RMV API reference and HAFAS documentation.
## Requirements # Requirements
* RMV API key (Get it [here](https://opendata.rmv.de/site/start.html)) * RMV API key (Get it [here](https://opendata.rmv.de/site/start.html))
* Python3 (Tested versions are 3.7.9 and 3.9.13) * Python3 (Tested versions are 3.7.9 and 3.9.13)
* git (Only for installation from source) * git (Only for installation from source)
## Installation # Installation
If you have everything listed in [requirements](#requirements), then let's begin. If you have everything listed in [requirements](#requirements), then let's begin.
### Variant 1 ### Variant 1:
1. `python -m pip install pyrmv` 1. `python -m pip install pyrmv`
### Variant 2 ### Variant 2:
1. `git clone https://git.end-play.xyz/profitroll/PythonRMV.git` 1. `git clone https://git.end-play.xyz/profitroll/PythonRMV.git`
2. `cd PythonRMV` 2. `cd PythonRMV`
3. `python setup.py install` 3. `python setup.py install`
## Usage # Usage
```py ```py
import pyrmv import pyrmv
@ -38,26 +36,44 @@ destination = client.stop_by_coords(50.099613, 8.685449, max_number=3)[0]
trip = client.trip_find(origin_id=origin.id, dest_id=destination.id) trip = client.trip_find(origin_id=origin.id, dest_id=destination.id)
``` ```
## Frequently Asked Questions # Frequently Asked Questions
* [Why are there raw versions and formatted ones?](#why-are-there-raw-versions-and-formatted-ones) - [Why are there raw versions and formatted ones?](#why-are-there-raw-versions-and-formatted-ones)
* [Some methods work slightly different](#some-methods-work-slightly-different) - [Some methods work slightly different](#some-methods-work-slightly-different)
### Why are there raw versions and formatted ones? ## Why are there raw versions and formatted ones?
For the purposes of my projects I don't really need all the stuff RMV gives (even though it's not much). For the purposes of my projects I don't really need all the stuff RMV gives (even though it's not much).
I only need some specific things. However I do understand that in some cases other users may find I only need some specific things. However I do understand that in some cases other users may find
those methods quite useful so I implemented them as well. those methods quite useful so I implemented them as well.
### Some methods work slightly different
## Some methods work slightly different
Can be. Not all function arguments written may work perfectly because I simply did not test each and Can be. Not all function arguments written may work perfectly because I simply did not test each and
every request. Some of arguments may be irrelevant in my use-case and the others are used quite rare at all. every request. Some of arguments may be irrelevant in my use-case and the others are used quite rare at all.
Just [make an issue](https://git.end-play.xyz/profitroll/PythonRMV/issues/new) and I'll implement it correct when I'll have some free time. Just [make an issue](https://git.end-play.xyz/profitroll/PythonRMV/issues/new) and I'll implement it correct when I'll have some free time.
## To-Do # To-Do
## General
- [ ] Docs in Wiki
### General ## Raw methods
- [x] arrivalBoard (board_arrival)
- [x] departureBoard (board_departure)
- [x] himsearch (him_search)
- [x] journeyDetail (journey_detail)
- [x] location.nearbystops (stop_by_coords)
- [x] location.name (stop_by_name)
- [x] trip (trip_find)
- [x] recon (trip_recon)
* [ ] Docs in Wiki ## Normal methods
* [ ] Tickets - [x] arrivalBoard (board_arrival)
- [x] departureBoard (board_departure)
- [x] himsearch (him_search)
- [x] journeyDetail (journey_detail)
- [x] location.nearbystops (stop_by_coords)
- [x] location.name (stop_by_name)
- [x] trip (trip_find)
- [x] recon (trip_recon)

View File

@ -70,15 +70,15 @@ class Client():
self.access_id = access_id self.access_id = access_id
def board_arrival(self, def board_arrival(self,
id: Union[str, None] = None, id: str = None, # type: ignore
id_ext: Union[str, None] = None, id_ext: str = None, # type: ignore
direction: Union[str, Stop, StopTrip, None] = None, direction: Union[str, Stop, StopTrip] = None, # type: ignore
date: Union[str, datetime, None] = None, date: Union[str, datetime] = None, # type: ignore
time: Union[str, datetime, None] = None, time: Union[str, datetime] = None, # type: ignore
duration: Union[int, timedelta] = 60, duration: Union[int, timedelta] = 60,
journeys_max: int = -1, journeys_max: int = -1,
operators: Union[str, list, None] = None, operators: Union[str, list] = None, # type: ignore
lines: Union[str, list, None] = None, lines: Union[str, list] = None, # type: ignore
passlist: bool = False, passlist: bool = False,
board_type: Literal[BoardArrivalType.ARR, BoardArrivalType.ARR_EQUIVS, BoardArrivalType.ARR_MAST, BoardArrivalType.ARR_STATION] = BoardArrivalType.ARR, board_type: Literal[BoardArrivalType.ARR, BoardArrivalType.ARR_EQUIVS, BoardArrivalType.ARR_MAST, BoardArrivalType.ARR_STATION] = BoardArrivalType.ARR,
) -> BoardArrival: ) -> BoardArrival:
@ -113,7 +113,7 @@ class Client():
direction=direction, # type: ignore direction=direction, # type: ignore
date=date, date=date,
time=time, time=time,
duration=duration, duration=duration, # type: ignore
maxJourneys=journeys_max, maxJourneys=journeys_max,
operators=operators, operators=operators,
lines=lines, lines=lines,
@ -126,15 +126,15 @@ class Client():
return BoardArrival(board_raw, self) return BoardArrival(board_raw, self)
def board_departure(self, def board_departure(self,
id: Union[str, None] = None, id: str = None, # type: ignore
id_ext: Union[str, None] = None, id_ext: str = None, # type: ignore
direction: Union[str, Stop, StopTrip, None] = None, direction: Union[str, Stop, StopTrip] = None, # type: ignore
date: Union[str, datetime, None] = None, date: Union[str, datetime] = None, # type: ignore
time: Union[str, datetime, None] = None, time: Union[str, datetime] = None, # type: ignore
duration: Union[int, timedelta] = 60, duration: Union[int, timedelta] = 60,
journeys_max: int = -1, journeys_max: int = -1,
operators: Union[str, list, None] = None, operators: Union[str, list] = None, # type: ignore
lines: Union[str, list, None] = None, lines: Union[str, list] = None, # type: ignore
passlist: bool = False, passlist: bool = False,
board_type: Literal[BoardDepartureType.DEP, BoardDepartureType.DEP_EQUIVS, BoardDepartureType.DEP_MAST, BoardDepartureType.DEP_STATION] = BoardDepartureType.DEP, board_type: Literal[BoardDepartureType.DEP, BoardDepartureType.DEP_EQUIVS, BoardDepartureType.DEP_MAST, BoardDepartureType.DEP_STATION] = BoardDepartureType.DEP,
) -> BoardDeparture: ) -> BoardDeparture:
@ -169,7 +169,7 @@ class Client():
direction=direction, # type: ignore direction=direction, # type: ignore
date=date, date=date,
time=time, time=time,
duration=duration, duration=duration, # type: ignore
maxJourneys=journeys_max, maxJourneys=journeys_max,
operators=operators, operators=operators,
lines=lines, lines=lines,
@ -182,28 +182,28 @@ class Client():
return BoardDeparture(board_raw, self) return BoardDeparture(board_raw, self)
def him_search(self, def him_search(self,
date_begin: Union[str, datetime, None] = None, date_begin: Union[str, datetime] = None, # type: ignore
date_end: Union[str, datetime, None] = None, date_end: Union[str, datetime] = None, # type: ignore
time_begin: Union[str, datetime, None] = None, time_begin: Union[str, datetime] = None, # type: ignore
time_end: Union[str, datetime, None] = None, time_end: Union[str, datetime] = None, # type: ignore
weekdays: Union[str, OrderedDict[str, bool], None] = None, weekdays: Union[str, OrderedDict[str, bool]] = None, # type: ignore
ids: Union[list, None] = None, ids: list = None, # type: ignore
operators: Union[list, None] = None, operators: list = None, # type: ignore
categories: Union[list, None] = None, categories: list = None, # type: ignore
channels: Union[list, None] = None, channels: list = None, # type: ignore
companies: Union[list, None] = None, companies: list = None, # type: ignore
lines: Union[list, None] = None, lines: list = None, # type: ignore
line_ids: Union[list, None] = None, line_ids: list = None, # type: ignore
stations: Union[list, List[Stop], None] = None, stations: Union[list, List[Stop]] = None, # type: ignore
station_from: Union[str, Stop, None] = None, station_from: Union[str, Stop] = None, # type: ignore
station_to: Union[str, Stop, None] = None, station_to: Union[str, Stop] = None, # type: ignore
both_ways: Union[bool, None] = None, both_ways: bool = None, # type: ignore
train_names: Union[list, None] = None, train_names: list = None, # type: ignore
search_mode: Union[Literal[SearchMode.MATCH, SearchMode.NOMATCH, SearchMode.TFMATCH], None] = None, search_mode: Literal[SearchMode.MATCH, SearchMode.NOMATCH, SearchMode.TFMATCH] = None, # type: ignore
affected_journey_mode: Union[Literal[AffectedJourneyMode.ALL, AffectedJourneyMode.OFF], None] = None, affected_journey_mode: Literal[AffectedJourneyMode.ALL, AffectedJourneyMode.OFF] = None, # type: ignore
affected_journey_stop_mode: Union[Literal[AffectedJourneyStopMode.ALL, AffectedJourneyStopMode.IMP, AffectedJourneyStopMode.OFF], None] = None, affected_journey_stop_mode: Literal[AffectedJourneyStopMode.ALL, AffectedJourneyStopMode.IMP, AffectedJourneyStopMode.OFF] = None, # type: ignore
priority_min: Union[int, None] = None, priority_min: int = None, # type: ignore
priority_max: Union[int, None] = None priority_max: int = None # type: ignore
) -> List[Message]: ) -> List[Message]:
"""The him_search method will deliver a list of HIM messages if matched by the given criteria as """The him_search method will deliver a list of HIM messages if matched by the given criteria as
well as affected products if any. well as affected products if any.
@ -239,10 +239,10 @@ class Client():
""" """
if isinstance(station_from, Stop): if isinstance(station_from, Stop):
station_from = station_from.ext_id station_from = station_from.ext_id # type: ignore
if isinstance(station_to, Stop): if isinstance(station_to, Stop):
station_to = station_to.ext_id station_to = station_to.ext_id # type: ignore
if stations != None: if stations != None:
new_stations = [] new_stations = []
@ -254,17 +254,17 @@ class Client():
stations = new_stations stations = new_stations
if search_mode == None: if search_mode == None:
search_mode = None search_mode = None # type: ignore
else: else:
search_mode = search_mode.code search_mode = search_mode.code
if affected_journey_mode == None: if affected_journey_mode == None:
affected_journey_mode = None affected_journey_mode = None # type: ignore
else: else:
affected_journey_mode = affected_journey_mode.code affected_journey_mode = affected_journey_mode.code
if affected_journey_stop_mode == None: if affected_journey_stop_mode == None:
affected_journey_stop_mode = None affected_journey_stop_mode = None # type: ignore
else: else:
affected_journey_stop_mode = affected_journey_stop_mode.code affected_journey_stop_mode = affected_journey_stop_mode.code
@ -304,12 +304,12 @@ class Client():
def journey_detail(self, def journey_detail(self,
id: str, id: str,
date: Union[str, datetime, None] = None, date: Union[str, datetime] = None, # type: ignore
real_time_mode: Union[Literal[RealTimeMode.FULL, RealTimeMode.INFOS, RealTimeMode.OFF, RealTimeMode.REALTIME, RealTimeMode.SERVER_DEFAULT], None] = None, real_time_mode: Literal[RealTimeMode.FULL, RealTimeMode.INFOS, RealTimeMode.OFF, RealTimeMode.REALTIME, RealTimeMode.SERVER_DEFAULT] = None, # type: ignore
from_id: Union[str, None] = None, from_id: str = None, # type: ignore
from_index: Union[int, None] = None, from_index: int = None, # type: ignore
to_id: Union[str, None] = None, to_id: str = None, # type: ignore
to_index: Union[int, None] = None to_index: int = None # type: ignore
) -> Journey: ) -> Journey:
"""The journey_detail method will deliver information about the complete route of a vehicle. The journey """The journey_detail method will deliver information about the complete route of a vehicle. The journey
identifier is part of a trip or `board_departure()` response. It contains a list of all stops/stations of this journey identifier is part of a trip or `board_departure()` response. It contains a list of all stops/stations of this journey
@ -332,7 +332,7 @@ class Client():
""" """
if real_time_mode == None: if real_time_mode == None:
real_time_mode = None real_time_mode = None # type: ignore
else: else:
real_time_mode = real_time_mode.code real_time_mode = real_time_mode.code
@ -342,9 +342,9 @@ class Client():
date=date, date=date,
rtMode=real_time_mode, # type: ignore rtMode=real_time_mode, # type: ignore
fromId=from_id, fromId=from_id,
fromIdx=from_index, fromIdx=from_index, # type: ignore
toId=to_id, toId=to_id,
toIdx=to_index toIdx=to_index # type: ignore
) )
find_exception(journey_raw) find_exception(journey_raw)
@ -359,7 +359,7 @@ class Client():
radius: Union[int, float] = 1000, radius: Union[int, float] = 1000,
max_number: int = 10, max_number: int = 10,
stop_type: Literal[LocationType.S, LocationType.P, LocationType.SP, LocationType.SE, LocationType.SPE] = LocationType.S, stop_type: Literal[LocationType.S, LocationType.P, LocationType.SP, LocationType.SE, LocationType.SPE] = LocationType.S,
selection_mode: Union[Literal[SelectionMode.SLCT_A, SelectionMode.SLCT_N], None] = None, selection_mode: Literal[SelectionMode.SLCT_A, SelectionMode.SLCT_N] = None, # type: ignore
) -> List[Stop]: ) -> List[Stop]:
"""Method returns a list of stops around a given center coordinate. """Method returns a list of stops around a given center coordinate.
The returned results are ordered by their distance to the center coordinate. The returned results are ordered by their distance to the center coordinate.
@ -380,7 +380,7 @@ class Client():
""" """
if selection_mode == None: if selection_mode == None:
selection_mode = None selection_mode = None # type: ignore
else: else:
selection_mode = selection_mode.code selection_mode = selection_mode.code
@ -446,12 +446,12 @@ class Client():
lang: Literal[Language.DE, Language.DA, Language.EN, Language.ES, Language.FR, Language.HU, Language.IT, Language.NL, Language.NO, Language.PL, Language.SV, Language.TR] = Language.EN, lang: Literal[Language.DE, Language.DA, Language.EN, Language.ES, Language.FR, Language.HU, Language.IT, Language.NL, Language.NO, Language.PL, Language.SV, Language.TR] = Language.EN,
max_number: int = 10, max_number: int = 10,
stop_type: Literal[LocationType.A, LocationType.ALL, LocationType.AP, LocationType.P, LocationType.S, LocationType.SA, LocationType.SP] = LocationType.ALL, stop_type: Literal[LocationType.A, LocationType.ALL, LocationType.AP, LocationType.P, LocationType.S, LocationType.SA, LocationType.SP] = LocationType.ALL,
selection_mode: Union[Literal[SelectionMode.SLCT_A, SelectionMode.SLCT_N], None] = None, selection_mode: Literal[SelectionMode.SLCT_A, SelectionMode.SLCT_N] = None, # type: ignore
coord_lat: Union[str, float, None] = None, coord_lat: Union[str, float] = None, # type: ignore
coord_lon: Union[str, float, None] = None, coord_lon: Union[str, float] = None, # type: ignore
radius: Union[int, float] = 1000, radius: Union[int, float] = 1000,
refine_id: Union[str, None] = None, refine_id: str = None, # type: ignore
stations: Union[str, list, None] = None, stations: Union[str, list] = None, # type: ignore
filter_mode: Literal[FilterMode.DIST_PERI, FilterMode.EXCL_PERI, FilterMode.SLCT_PERI] = FilterMode.DIST_PERI filter_mode: Literal[FilterMode.DIST_PERI, FilterMode.EXCL_PERI, FilterMode.SLCT_PERI] = FilterMode.DIST_PERI
) -> List[Stop]: ) -> List[Stop]:
"""Method can be used to perform a pattern matching of a user input and to retrieve a list """Method can be used to perform a pattern matching of a user input and to retrieve a list
@ -478,7 +478,7 @@ class Client():
""" """
if selection_mode == None: if selection_mode == None:
selection_mode = None selection_mode = None # type: ignore
else: else:
selection_mode = selection_mode.code selection_mode = selection_mode.code
@ -511,47 +511,47 @@ class Client():
def trip_find(self, def trip_find(self,
lang: Literal[Language.DE, Language.DA, Language.EN, Language.ES, Language.FR, Language.HU, Language.IT, Language.NL, Language.NO, Language.PL, Language.SV, Language.TR] = Language.EN, lang: Literal[Language.DE, Language.DA, Language.EN, Language.ES, Language.FR, Language.HU, Language.IT, Language.NL, Language.NO, Language.PL, Language.SV, Language.TR] = Language.EN,
origin_id: Union[str, None] = None, origin_id: str = None, # type: ignore
origin_id_ext: Union[str, None] = None, origin_id_ext: str = None, # type: ignore
origin_coord_lat: Union[str, float, None] = None, origin_coord_lat: Union[str, float] = None, # type: ignore
origin_coord_lon: Union[str, float, None] = None, origin_coord_lon: Union[str, float] = None, # type: ignore
origin_coord_name: Union[str, None] = None, origin_coord_name: str = None, # type: ignore
destination_id: Union[str, None] = None, destination_id: str = None, # type: ignore
destination_id_ext: Union[str, None] = None, destination_id_ext: str = None, # type: ignore
destination_coord_lat: Union[str, float, None] = None, destination_coord_lat: Union[str, float] = None, # type: ignore
destination_coord_lon: Union[str, float, None] = None, destination_coord_lon: Union[str, float] = None, # type: ignore
destination_coord_name: Union[str, None] = None, destination_coord_name: str = None, # type: ignore
via: Union[str, None] = None, via: str = None, # type: ignore
via_id: Union[str, None] = None, via_id: str = None, # type: ignore
via_gis: Union[str, None] = None, via_gis: str = None, # type: ignore
via_wait_time: int = 0, via_wait_time: int = 0,
avoid: Union[str, None] = None, avoid: str = None, # type: ignore
avoid_id: Union[str, None] = None, avoid_id: str = None, # type: ignore
change_time_percent: int = 100, change_time_percent: int = 100,
change_time_min: Union[int, None] = None, change_time_min: int = None, # type: ignore
change_time_max: Union[int, None] = None, change_time_max: int = None, # type: ignore
change_time_add: Union[int, None] = None, change_time_add: int = None, # type: ignore
change_max: Union[int, None] = None, change_max: int = None, # type: ignore
date: Union[str, datetime, None] = None, date: Union[str, datetime] = None, # type: ignore
time: Union[str, datetime, None] = None, time: Union[str, datetime] = None, # type: ignore
search_arrival: bool = False, search_arrival: bool = False,
trips_after_time: Union[int, None] = None, trips_after_time: int = None, # type: ignore
trips_before_time: Union[int, None] = None, trips_before_time: int = None, # type: ignore
context: Union[str, None] = None, context: str = None, # type: ignore
passlist: bool = False, passlist: bool = False,
operators: Union[str, list, None] = None, operators: Union[str, list] = None, # type: ignore
lines: Union[str, list, None] = None, lines: Union[str, list] = None, # type: ignore
lineids: Union[str, list, None] = None, lineids: Union[str, list] = None, # type: ignore
iv_include: bool = False, iv_include: bool = False,
iv_only: bool = False, iv_only: bool = False,
@ -560,11 +560,11 @@ class Client():
passing_points: bool = False, passing_points: bool = False,
real_time_mode: Union[Literal[RealTimeMode.FULL, RealTimeMode.INFOS, RealTimeMode.OFF, RealTimeMode.REALTIME, RealTimeMode.SERVER_DEFAULT], None] = None, real_time_mode: Literal[RealTimeMode.FULL, RealTimeMode.INFOS, RealTimeMode.OFF, RealTimeMode.REALTIME, RealTimeMode.SERVER_DEFAULT] = None, # type: ignore
include_earlier: bool = False, include_earlier: bool = False,
ict_alternatives: bool = False, ict_alternatives: bool = False,
tariff: Union[bool, None] = None, tariff: bool = None, # type: ignore
messages: bool = False, messages: bool = False,
frequency: bool = True frequency: bool = True
) -> List[Trip]: ) -> List[Trip]:
@ -623,7 +623,7 @@ class Client():
""" """
if real_time_mode == None: if real_time_mode == None:
real_time_mode = None real_time_mode = None # type: ignore
else: else:
real_time_mode = real_time_mode.code real_time_mode = real_time_mode.code
@ -700,17 +700,17 @@ class Client():
def trip_recon(self, def trip_recon(self,
context: Union[str, Trip], context: Union[str, Trip],
date: Union[str, datetime, None] = None, date: Union[str, datetime] = None, # type: ignore
match_real_time: Union[bool, None] = None, match_real_time: bool = None, # type: ignore
enable_replacements: Union[bool, None] = None, enable_replacements: bool = None, # type: ignore
arrival_dev_lower: Union[int, None] = None, arrival_dev_lower: int = None, # type: ignore
arrival_dev_upper: Union[int, None] = None, arrival_dev_upper: int = None, # type: ignore
departure_dev_lower: Union[int, None] = None, departure_dev_lower: int = None, # type: ignore
departure_dev_upper: Union[int, None] = None, departure_dev_upper: int = None, # type: ignore
passlist: bool = False, passlist: bool = None, # type: ignore
passing_points: bool = False, passing_points: bool = False,
real_time_mode: Union[Literal[RealTimeMode.FULL, RealTimeMode.INFOS, RealTimeMode.OFF, RealTimeMode.REALTIME, RealTimeMode.SERVER_DEFAULT], None] = None, real_time_mode: Literal[RealTimeMode.FULL, RealTimeMode.INFOS, RealTimeMode.OFF, RealTimeMode.REALTIME, RealTimeMode.SERVER_DEFAULT] = None, # type: ignore
tariff: Union[bool, None] = None, tariff: bool = None, # type: ignore
messages: bool = False messages: bool = False
) -> List[Trip]: ) -> List[Trip]:
"""Reconstructing a trip can be achieved using the reconstruction context provided by any trip result in the """Reconstructing a trip can be achieved using the reconstruction context provided by any trip result in the
@ -739,7 +739,7 @@ class Client():
""" """
if real_time_mode == None: if real_time_mode == None:
real_time_mode = None real_time_mode = None # type: ignore
else: else:
real_time_mode = real_time_mode.code real_time_mode = real_time_mode.code

View File

@ -11,19 +11,19 @@ except ImportError:
# 2.25. Arrival Board (arrivalBoard) # 2.25. Arrival Board (arrivalBoard)
def board_arrival(accessId: str, def board_arrival(accessId: str,
json: bool = True, json: bool = True,
id: Union[str, None] = None, id: str = None, # type: ignore
extId: Union[str, None] = None, extId: str = None, # type: ignore
direction: Union[str, None] = None, direction: str = None, # type: ignore
date: Union[str, datetime, None] = None, date: Union[str, datetime] = None, # type: ignore
time: Union[str, datetime, None] = None, time: Union[str, datetime] = None, # type: ignore
duration: Union[int, timedelta] = 60, duration: int = 60,
maxJourneys: int = -1, maxJourneys: int = -1,
products: Union[int, None] = None, products: int = None, # type: ignore
operators: Union[str, list, None] = None, operators: Union[str, list] = None, # type: ignore
lines: Union[str, list, None] = None, lines: Union[str, list] = None, # type: ignore
filterEquiv: bool = True, filterEquiv: bool = True,
attributes: Union[str, list, None] = None, attributes: Union[str, list] = None, # type: ignore
platforms: Union[str, list, None] = None, platforms: Union[str, list] = None, # type: ignore
passlist: bool = False, passlist: bool = False,
boardType: Literal["ARR", "ARR_EQUIVS", "ARR_MAST", "ARR_STATION"] = "ARR" boardType: Literal["ARR", "ARR_EQUIVS", "ARR_MAST", "ARR_STATION"] = "ARR"
) -> dict: ) -> dict:
@ -81,7 +81,7 @@ def board_arrival(accessId: str,
elif str(var) == "duration": elif str(var) == "duration":
if val != None: if val != None:
if isinstance(val, timedelta): if isinstance(val, timedelta):
payload[str(var)] = val.minutes # type: ignore payload[str(var)] = val.minutes
else: else:
payload[str(var)] = val payload[str(var)] = val
elif str(var) == "boardType": elif str(var) == "boardType":

View File

@ -11,19 +11,19 @@ except ImportError:
# 2.24. Departure Board (departureBoard) # 2.24. Departure Board (departureBoard)
def board_departure(accessId: str, def board_departure(accessId: str,
json: bool = True, json: bool = True,
id: Union[str, None] = None, id: str = None, # type: ignore
extId: Union[str, None] = None, extId: str = None, # type: ignore
direction: Union[str, None] = None, direction: str = None, # type: ignore
date: Union[str, datetime, None] = None, date: Union[str, datetime] = None, # type: ignore
time: Union[str, datetime, None] = None, time: Union[str, datetime] = None, # type: ignore
duration: Union[int, timedelta] = 60, duration: int = 60,
maxJourneys: int = -1, maxJourneys: int = -1,
products: Union[int, None] = None, products: int = None, # type: ignore
operators: Union[str, list, None] = None, operators: Union[str, list] = None, # type: ignore
lines: Union[str, list, None] = None, lines: Union[str, list] = None, # type: ignore
filterEquiv: bool = True, filterEquiv: bool = True,
attributes: Union[str, list, None] = None, attributes: Union[str, list] = None, # type: ignore
platforms: Union[str, list, None] = None, platforms: Union[str, list] = None, # type: ignore
passlist: bool = False, passlist: bool = False,
boardType: Literal["DEP", "DEP_EQUIVS", "DEP_MAST", "DEP_STATION"] = "DEP" boardType: Literal["DEP", "DEP_EQUIVS", "DEP_MAST", "DEP_STATION"] = "DEP"
) -> dict: ) -> dict:
@ -82,7 +82,7 @@ def board_departure(accessId: str,
elif str(var) == "duration": elif str(var) == "duration":
if val != None: if val != None:
if isinstance(val, timedelta): if isinstance(val, timedelta):
payload[str(var)] = val.minutes # type: ignore payload[str(var)] = val.minutes
else: else:
payload[str(var)] = val payload[str(var)] = val
elif str(var) == "boardType": elif str(var) == "boardType":

View File

@ -13,38 +13,38 @@ except ImportError:
# 2.37. HIM Search (himsearch) # 2.37. HIM Search (himsearch)
def him_search(accessId: str, def him_search(accessId: str,
json: bool = True, json: bool = True,
dateB: Union[str, datetime, None] = None, dateB: Union[str, datetime] = None, # type: ignore
dateE: Union[str, datetime, None] = None, dateE: Union[str, datetime] = None, # type: ignore
timeB: Union[str, datetime, None] = None, timeB: Union[str, datetime] = None, # type: ignore
timeE: Union[str, datetime, None] = None, timeE: Union[str, datetime] = None, # type: ignore
weekdays: Union[str, OrderedDict[str, bool], None] = None, weekdays: Union[str, OrderedDict[str, bool]] = None, # type: ignore
himIds: Union[str, list, None] = None, himIds: Union[str, list] = None, # type: ignore
hierarchicalView: bool = False, hierarchicalView: bool = False,
operators: Union[str, list, None] = None, operators: Union[str, list] = None, # type: ignore
categories: Union[str, list, None] = None, categories: Union[str, list] = None, # type: ignore
channels: Union[str, list, None] = None, channels: Union[str, list] = None, # type: ignore
companies: Union[str, list, None] = None, companies: Union[str, list] = None, # type: ignore
lines: Union[str, list, None] = None, lines: Union[str, list] = None, # type: ignore
lineids: Union[str, list, None] = None, lineids: Union[str, list] = None, # type: ignore
stations: Union[str, list, None] = None, stations: Union[str, list] = None, # type: ignore
fromstation: Union[str, None] = None, fromstation: str = None, # type: ignore
tostation: Union[str, None] = None, tostation: str = None, # type: ignore
bothways: Union[bool, None] = None, bothways: bool = None, # type: ignore
trainnames: Union[str, list, None] = None, trainnames: Union[str, list] = None, # type: ignore
metas: Union[str, list, None] = None, metas: Union[str, list] = None, # type: ignore
himcategory: Union[str, None] = None, himcategory: str = None, # type: ignore
himtags: Union[str, list, None] = None, himtags: Union[str, list] = None, # type: ignore
regions: Union[str, list, None] = None, regions: Union[str, list] = None, # type: ignore
himtext: Union[str, list, None] = None, himtext: Union[str, list] = None, # type: ignore
himtexttags: Union[str, list, None] = None, himtexttags: Union[str, list] = None, # type: ignore
additionalfields: Union[str, list, dict, None] = None, additionalfields: Union[str, list, dict] = None, # type: ignore
poly: bool = False, poly: bool = False,
searchmode: Union[Literal["MATCH", "NOMATCH", "TFMATCH"], None] = None, searchmode: Literal["MATCH", "NOMATCH", "TFMATCH"] = None, # type: ignore
affectedJourneyMode: Union[Literal["ALL", "OFF"], None] = None, affectedJourneyMode: Literal["ALL", "OFF"] = None, # type: ignore
affectedJourneyStopMode: Union[Literal["ALL", "IMP", "OFF"], None] = None, affectedJourneyStopMode: Literal["ALL", "IMP", "OFF"] = None, # type: ignore
orderBy: Union[str, list, None] = None, orderBy: Union[str, list] = None, # type: ignore
minprio: Union[str, int, None] = None, minprio: Union[str, int] = None, # type: ignore
maxprio: Union[str, int, None] = None maxprio: Union[str, int] = None # type: ignore
) -> dict: ) -> dict:
"""The himSearch will return a list of HIM messages if matched by the given criteria as well as affected """The himSearch will return a list of HIM messages if matched by the given criteria as well as affected
products if any. products if any.

View File

@ -12,15 +12,15 @@ except ImportError:
def journey_detail(accessId: str, def journey_detail(accessId: str,
id: str, id: str,
json: bool = True, json: bool = True,
date: Union[str, datetime, None] = None, date: Union[str, datetime] = None, # type: ignore
poly: bool = False, poly: bool = False,
polyEnc: Literal["DLT", "GPA", "N"] = "N", polyEnc: Literal["DLT", "GPA", "N"] = "N",
showPassingPoints: bool = False, showPassingPoints: bool = False,
rtMode: Union[Literal["FULL", "INFOS", "OFF", "REALTIME", "SERVER_DEFAULT"], None] = None, rtMode: Literal["FULL", "INFOS", "OFF", "REALTIME", "SERVER_DEFAULT"] = None, # type: ignore
fromId: Union[str, None] = None, fromId: str = None, # type: ignore
fromIdx: Union[int, None] = None, fromIdx: str = None, # type: ignore
toId: Union[str, None] = None, toId: str = None, # type: ignore
toIdx: Union[int, None] = None, toIdx: str = None, # type: ignore
baim: bool = False baim: bool = False
) -> dict: ) -> dict:
"""The journey_detail method will deliver information about the complete route of a vehicle. The journey """The journey_detail method will deliver information about the complete route of a vehicle. The journey

View File

@ -16,11 +16,11 @@ def stop_by_coords(accessId: str,
radius: Union[int, float] = 1000, radius: Union[int, float] = 1000,
maxNo: int = 10, maxNo: int = 10,
stopType: Literal["S", "P", "SP", "SE", "SPE"] = "S", stopType: Literal["S", "P", "SP", "SE", "SPE"] = "S",
locationSelectionMode: Union[Literal["SLCT_N", "SLCT_A"], None] = None, locationSelectionMode: Literal["SLCT_N", "SLCT_A"] = None, # type: ignore
products: Union[int, None] = None, products: int = None, # type: ignore
meta: Union[str, None] = None, meta: str = None, # type: ignore
sattributes: Union[str, list, None] = None, sattributes: Union[str, list] = None, # type: ignore
sinfotexts: Union[str, list, None] = None sinfotexts: Union[str, list] = None # type: ignore
) -> dict: ) -> dict:
"""The location.nearbystops service returns a list of stops around a given center coordinate (within a """The location.nearbystops service returns a list of stops around a given center coordinate (within a
radius of 1000m). The returned results are ordered by their distance to the center coordinate. radius of 1000m). The returned results are ordered by their distance to the center coordinate.

View File

@ -14,15 +14,15 @@ def stop_by_name(accessId: str,
json: bool = True, json: bool = True,
maxNo: int = 10, maxNo: int = 10,
stopType: Literal["A", "ALL", "AP", "P", "S", "SA", "SP"] = "ALL", stopType: Literal["A", "ALL", "AP", "P", "S", "SA", "SP"] = "ALL",
locationSelectionMode: Union[Literal["SLCT_N", "SLCT_A"], None] = None, locationSelectionMode: Literal["SLCT_N", "SLCT_A"] = None, # type: ignore
products: Union[int, None] = None, products: int = None, # type: ignore
coordLat: Union[str, float, None] = None, coordLat: Union[str, float] = None, # type: ignore
coordLong: Union[str, float, None] = None, coordLong: Union[str, float] = None, # type: ignore
radius: Union[int, float] = 1000, radius: Union[int, float] = 1000,
refineId: Union[str, None] = None, refineId: str = None, # type: ignore
meta: Union[str, None] = None, meta: str = None, # type: ignore
stations: Union[str, list, None] = None, stations: Union[str, list] = None, # type: ignore
sattributes: Union[str, list, None] = None, sattributes: Union[str, list] = None, # type: ignore
filterMode: Literal["DIST_PERI", "EXCL_PERI", "SLCT_PERI"] = "DIST_PERI" filterMode: Literal["DIST_PERI", "EXCL_PERI", "SLCT_PERI"] = "DIST_PERI"
) -> dict: ) -> dict:
"""The location.name service can be used to perform a pattern matching of a user input and to retrieve a list """The location.name service can be used to perform a pattern matching of a user input and to retrieve a list

View File

@ -13,87 +13,87 @@ def trip_find(accessId: str,
lang: Literal["de", "da", "en", "es", "fr", "hu", "it", "nl", "no", "pl", "sv", "tr"] = "en", lang: Literal["de", "da", "en", "es", "fr", "hu", "it", "nl", "no", "pl", "sv", "tr"] = "en",
json: bool = True, json: bool = True,
originId: Union[str, None] = None, originId: str = None, # type: ignore
originExtId: Union[str, None] = None, originExtId: str = None, # type: ignore
originCoordLat: Union[str, float, None] = None, originCoordLat: Union[str, float] = None, # type: ignore
originCoordLong: Union[str, float, None] = None, originCoordLong: Union[str, float] = None, # type: ignore
originCoordName: Union[str, None] = None, originCoordName: str = None, # type: ignore
destId: Union[str, None] = None, destId: str = None, # type: ignore
destExtId: Union[str, None] = None, destExtId: str = None, # type: ignore
destCoordLat: Union[str, float, None] = None, destCoordLat: Union[str, float] = None, # type: ignore
destCoordLong: Union[str, float, None] = None, destCoordLong: Union[str, float] = None, # type: ignore
destCoordName: Union[str, None] = None, destCoordName: str = None, # type: ignore
via: Union[str, None] = None, via: str = None, # type: ignore
viaId: Union[str, None] = None, viaId: str = None, # type: ignore
viaWaitTime: int = 0, viaWaitTime: int = 0,
avoid: Union[str, None] = None, avoid: str = None, # type: ignore
avoidId: Union[str, None] = None, avoidId: str = None, # type: ignore
viaGis: Union[str, None] = None, viaGis: str = None, # type: ignore
changeTimePercent: int = 100, changeTimePercent: int = 100,
minChangeTime: Union[int, None] = None, minChangeTime: int = None, # type: ignore
maxChangeTime: Union[int, None] = None, maxChangeTime: int = None, # type: ignore
addChangeTime: Union[int, None] = None, addChangeTime: int = None, # type: ignore
maxChange: Union[int, None] = None, maxChange: int = None, # type: ignore
date: Union[str, datetime, None] = None, date: Union[str, datetime] = None, # type: ignore
time: Union[str, datetime, None] = None, time: Union[str, datetime] = None, # type: ignore
searchForArrival: bool = False, searchForArrival: bool = False,
numF: Union[int, None] = None, numF: int = None, # type: ignore
numB: Union[int, None] = None, numB: int = None, # type: ignore
context: Union[str, None] = None, context: str = None, # type: ignore
poly: bool = False, poly: bool = False,
polyEnc: Literal["DLT", "GPA", "N"] = "N", polyEnc: Literal["DLT", "GPA", "N"] = "N",
passlist: bool = False, passlist: bool = False,
products: Union[int, None] = None, products: int = None, # type: ignore
operators: Union[str, list, None] = None, operators: Union[str, list] = None, # type: ignore
attributes: Union[str, list, None] = None, attributes: Union[str, list] = None, # type: ignore
sattributes: Union[str, list, None] = None, sattributes: Union[str, list] = None, # type: ignore
fattributes: Union[str, list, None] = None, fattributes: Union[str, list] = None, # type: ignore
lines: Union[str, list, None] = None, lines: Union[str, list] = None, # type: ignore
lineids: Union[str, list, None] = None, lineids: Union[str, list] = None, # type: ignore
avoidPaths: Union[List[Literal["SW", "EA", "ES", "RA", "CB"]], None] = None, avoidPaths: List[Literal["SW", "EA", "ES", "RA", "CB"]] = None, # type: ignore
originWalk: Union[str, list, None] = None, originWalk: Union[str, list] = None, # type: ignore
originBike: Union[str, list, None] = None, originBike: Union[str, list] = None, # type: ignore
originCar: Union[str, list, None] = None, originCar: Union[str, list] = None, # type: ignore
originTaxi: Union[str, list, None] = None, originTaxi: Union[str, list] = None, # type: ignore
originPark: Union[str, list, None] = None, originPark: Union[str, list] = None, # type: ignore
originMeta: Union[str, list, None] = None, originMeta: Union[str, list] = None, # type: ignore
destWalk: Union[str, list, None] = None, destWalk: Union[str, list] = None, # type: ignore
destBike: Union[str, list, None] = None, destBike: Union[str, list] = None, # type: ignore
destCar: Union[str, list, None] = None, destCar: Union[str, list] = None, # type: ignore
destTaxi: Union[str, list, None] = None, destTaxi: Union[str, list] = None, # type: ignore
destPark: Union[str, list, None] = None, destPark: Union[str, list] = None, # type: ignore
destMeta: Union[str, list, None] = None, destMeta: Union[str, list] = None, # type: ignore
totalWalk: Union[str, list, None] = None, totalWalk: Union[str, list] = None, # type: ignore
totalBike: Union[str, list, None] = None, totalBike: Union[str, list] = None, # type: ignore
totalCar: Union[str, list, None] = None, totalCar: Union[str, list] = None, # type: ignore
totalTaxi: Union[str, list, None] = None, totalTaxi: Union[str, list] = None, # type: ignore
totalMeta: Union[str, list, None] = None, totalMeta: Union[str, list] = None, # type: ignore
gisProducts: Union[str, None] = None, gisProducts: str = None, # type: ignore
includeIv: bool = False, includeIv: bool = False,
ivOnly: bool = False, ivOnly: bool = False,
mobilityProfile: Union[str, None] = None, mobilityProfile: str = None, # type: ignore
bikeCarriage: bool = False, bikeCarriage: bool = False,
bikeCarriageType: Union[Literal["SINGLEBIKES", "SMALLGROUPS", "LARGEGROUPS"], None] = None, bikeCarriageType: Literal["SINGLEBIKES", "SMALLGROUPS", "LARGEGROUPS"] = None, # type: ignore
sleepingCar: bool = False, sleepingCar: bool = False,
couchetteCoach: bool = False, couchetteCoach: bool = False,
@ -102,24 +102,24 @@ def trip_find(accessId: str,
eco: bool = False, eco: bool = False,
ecoCmp: bool = False, ecoCmp: bool = False,
ecoParams: Union[str, None] = None, ecoParams: str = None, # type: ignore
rtMode: Union[Literal["FULL", "INFOS", "OFF", "REALTIME", "SERVER_DEFAULT"], None] = None, rtMode: Literal["FULL", "INFOS", "OFF", "REALTIME", "SERVER_DEFAULT"] = None, # type: ignore
unsharp: bool = False, unsharp: bool = False,
trainFilter: Union[str, None] = None, trainFilter: str = None, # type: ignore
economic: bool = False, economic: bool = False,
groupFilter: Union[str, None] = None, groupFilter: str = None, # type: ignore
blockingList: Union[str, None] = None, blockingList: str = None, # type: ignore
blockedEdges: Union[str, None] = None, blockedEdges: str = None, # type: ignore
trainComposition: bool = False, trainComposition: bool = False,
includeEarlier: bool = False, includeEarlier: bool = False,
withICTAlternatives: bool = False, withICTAlternatives: bool = False,
tariff: Union[bool, None] = None, tariff: bool = None, # type: ignore
trafficMessages: bool = False, trafficMessages: bool = False,
travellerProfileData: Union[str, None] = None, travellerProfileData: str = None, # type: ignore
withFreq: bool = True withFreq: bool = True
) -> dict: ) -> dict:
"""The trip service calculates a trip from a specified origin to a specified destination. These might be """The trip service calculates a trip from a specified origin to a specified destination. These might be

View File

@ -14,31 +14,31 @@ def trip_recon(accessId: str,
json: bool = True, json: bool = True,
poly: bool = False, poly: bool = False,
polyEnc: Literal["DLT", "GPA", "N"] = "N", polyEnc: Literal["DLT", "GPA", "N"] = "N",
date: Union[str, datetime, None] = None, date: Union[str, datetime] = None, # type: ignore
useCombinedComparison: Union[bool, None] = None, useCombinedComparison: bool = None, # type: ignore
acceptGaps: Union[bool, None] = None, acceptGaps: bool = None, # type: ignore
allowDummySections: Union[bool, None] = None, allowDummySections: bool = None, # type: ignore
flagAllNonReachable: Union[bool, None] = None, flagAllNonReachable: bool = None, # type: ignore
matchCatStrict: Union[bool, None] = None, matchCatStrict: bool = None, # type: ignore
matchIdNonBlank: Union[bool, None] = None, matchIdNonBlank: bool = None, # type: ignore
matchIdStrict: Union[bool, None] = None, matchIdStrict: bool = None, # type: ignore
matchNumStrict: Union[bool, None] = None, matchNumStrict: bool = None, # type: ignore
matchRtType: Union[bool, None] = None, matchRtType: bool = None, # type: ignore
enableRtFullSearch: Union[bool, None] = None, enableRtFullSearch: bool = None, # type: ignore
enableReplacements: Union[bool, None] = None, enableReplacements: bool = None, # type: ignore
arrL: Union[int, None] = None, arrL: int = None, # type: ignore
arrU: Union[int, None] = None, arrU: int = None, # type: ignore
depL: Union[int, None] = None, depL: int = None, # type: ignore
depU: Union[int, None] = None, depU: int = None, # type: ignore
passlist: bool = False, passlist: bool = False,
showPassingPoints: bool = False, showPassingPoints: bool = False,
rtMode: Union[Literal["FULL", "INFOS", "OFF", "REALTIME", "SERVER_DEFAULT"], None] = None, rtMode: Literal["FULL", "INFOS", "OFF", "REALTIME", "SERVER_DEFAULT"] = None, # type: ignore
eco: bool = False, eco: bool = False,
ecoCmp: bool = False, ecoCmp: bool = False,
ecoParams: Union[str, None] = None, ecoParams: str = None, # type: ignore
tariff: Union[bool, None] = None, tariff: bool = None, # type: ignore
trafficMessages: Union[bool, None] = None, trafficMessages: bool = None, # type: ignore
travellerProfileData: Union[str, None] = None travellerProfileData: str = None # type: ignore
) -> dict: ) -> dict:
"""Reconstructing a trip can be achieved using the reconstruction context provided by any trip result in the """Reconstructing a trip can be achieved using the reconstruction context provided by any trip result in the
ctxRecon attribute of Trip element. The result will be a true copy of the original trip search result given ctxRecon attribute of Trip element. The result will be a true copy of the original trip search result given

View File

@ -5,7 +5,7 @@ setup(
version="0.3.1", version="0.3.1",
author="Profitroll", author="Profitroll",
description="Small module that makes your journey with RMV REST API somehow easier.", description="Small module that makes your journey with RMV REST API somehow easier.",
long_description="Small module that makes your journey with RMV REST API somehow easier. Based fully on official RMV API reference and HAFAS documentation.\n\n## Requirements\n\n* RMV API key (Get it [here](https://opendata.rmv.de/site/start.html))\n* Python3 (Tested versions are 3.7.9 and 3.9.13)\n* git (Only for installation from source)\n\n## Installation\n\nIf you have everything listed in [requirements](#requirements), then let's begin.\n\n### Variant 1\n\n1. `python -m pip install pyrmv`\n\n### Variant 2\n\n1. `git clone https://git.end-play.xyz/profitroll/PythonRMV.git`\n2. `cd PythonRMV`\n3. `python setup.py install`\n\n## Usage\n\n```py\nimport pyrmv\n\n# Define a Client with API key\nclient = pyrmv.Client(\"AcessId\")\n\n# Get origin's and destination's location\norigin = client.stop_by_name(\"Frankfurt Hauptbahnhof\", max_number=3)[0]\ndestination = client.stop_by_coords(50.099613, 8.685449, max_number=3)[0]\n\n# Find a trip by locations got\ntrip = client.trip_find(origin_id=origin.id, dest_id=destination.id)\n```\n\n## Frequently Asked Questions\n\n* [Why are there raw versions and formatted ones?](#why-are-there-raw-versions-and-formatted-ones)\n* [Some methods work slightly different](#some-methods-work-slightly-different)\n\n### Why are there raw versions and formatted ones?\n\nFor the purposes of my projects I don't really need all the stuff RMV gives (even though it's not much).\nI only need some specific things. However I do understand that in some cases other users may find\nthose methods quite useful so I implemented them as well.\n\n### Some methods work slightly different\n\nCan be. Not all function arguments written may work perfectly because I simply did not test each and\nevery request. Some of arguments may be irrelevant in my use-case and the others are used quite rare at all.\nJust [make an issue](https://git.end-play.xyz/profitroll/PythonRMV/issues/new) and I'll implement it correct when I'll have some free time.\n\n## To-Do\n\n### General\n\n* [ ] Docs in Wiki\n* [ ] Tickets", long_description="Small module that makes your journey with RMV REST API somehow easier. Based fully on official RMV API reference and HAFAS documentation.\n\n# Usage\n\n```py\nimport pyrmv\n\n# Define a Client with API key\nclient = pyrmv.Client(\"AcessId\")\n\n# Get origin's and destination's location\norigin = client.stop_by_name(\"Frankfurt Hauptbahnhof\", max_number=3)[0]\ndestination = client.stop_by_coords(50.099613, 8.685449, max_number=3)[0]\n\n# Find a trip by locations got\ntrip = client.trip_find(origin_id=origin.id, dest_id=destination.id)\n```\n\n# Frequently Asked Questions\n\n- Why are there raw versions and formatted ones?\n- Some methods work slightly different\n\n## Why are there raw versions and formatted ones?\n\nFor the purposes of my projects I don't really need all the stuff RMV gives (even though it's not much).\nI only need some specific things. However I do understand that in some cases other users may find\nthose methods quite useful so I implemented them as well.\n\n\n## Some methods work slightly different\n\nCan be. Not all function arguments written may work perfectly because I simply did not test each and\nevery request. Some of arguments may be irrelevant in my use-case and the others are used quite rare at all.\nJust [make an issue](https://git.end-play.xyz/profitroll/PythonRMV/issues/new) and I'll implement it correct when I'll have some free time.\n\n# To-Do\n## General\n- [ ] Docs in Wiki\n\n## Raw methods\n- [x] arrivalBoard (board_arrival) \n- [x] departureBoard (board_departure) \n- [x] himsearch (him_search) \n- [x] journeyDetail (journey_detail)\n- [x] location.nearbystops (stop_by_coords) \n- [x] location.name (stop_by_name) \n- [x] trip (trip_find) \n- [x] recon (trip_recon)\n\n## Normal methods\n- [x] arrivalBoard (board_arrival) \n- [x] departureBoard (board_departure) \n- [x] himsearch (him_search) \n- [x] journeyDetail (journey_detail)\n- [x] location.nearbystops (stop_by_coords) \n- [x] location.name (stop_by_name) \n- [x] trip (trip_find) \n- [x] recon (trip_recon)",
long_description_content_type="text/markdown", long_description_content_type="text/markdown",
author_email="profitroll@end-play.xyz", author_email="profitroll@end-play.xyz",
url="https://git.end-play.xyz/profitroll/PythonRMV", url="https://git.end-play.xyz/profitroll/PythonRMV",