mirror of
https://github.com/home-assistant/core.git
synced 2025-04-24 01:08:12 +00:00
Add london_underground (#8272)
* Add tube_state Add tube_state sensor * Final cleanup * Make corrections Correct PLATFORM_SCHEMA * Fix space * Make test pass * Correct format of test Test still failing, don’t understand why * correct description * Make test pass Preferred method below returns None state = self.hass.states.get('sensor.london_overground') * Format for hound * indent * Make requested changes to test, not working Test fails with: AssertionError: assert 0 > 0 where 0 = len([]) Surely I need tube_state.setup_platform ? * Fixed test Config was wrong * Change component name to london_tube * Update name to london_underground Make consistent * cleanup
This commit is contained in:
parent
05ced33648
commit
865865ca0f
135
homeassistant/components/sensor/london_underground.py
Normal file
135
homeassistant/components/sensor/london_underground.py
Normal file
@ -0,0 +1,135 @@
|
||||
"""
|
||||
Sensor for checking the status of London Underground tube lines.
|
||||
|
||||
For more details about this component, please refer to the documentation at
|
||||
https://home-assistant.io/components/sensor.tube-state
|
||||
"""
|
||||
import logging
|
||||
from datetime import timedelta
|
||||
|
||||
import voluptuous as vol
|
||||
import requests
|
||||
|
||||
import homeassistant.helpers.config_validation as cv
|
||||
from homeassistant.components.sensor import PLATFORM_SCHEMA
|
||||
from homeassistant.helpers.entity import Entity
|
||||
from homeassistant.util import Throttle
|
||||
|
||||
_LOGGER = logging.getLogger(__name__)
|
||||
|
||||
ATTRIBUTION = "Powered by TfL Open Data"
|
||||
CONF_LINE = 'line'
|
||||
SCAN_INTERVAL = timedelta(seconds=30)
|
||||
TUBE_LINES = [
|
||||
'Bakerloo',
|
||||
'Central',
|
||||
'Circle',
|
||||
'District',
|
||||
'DLR',
|
||||
'Hammersmith & City',
|
||||
'Jubilee',
|
||||
'London Overground',
|
||||
'Metropolitan',
|
||||
'Northern',
|
||||
'Piccadilly',
|
||||
'TfL Rail',
|
||||
'Victoria',
|
||||
'Waterloo & City']
|
||||
URL = 'https://api.tfl.gov.uk/line/mode/tube,overground,dlr,tflrail/status'
|
||||
|
||||
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
||||
vol.Required(CONF_LINE):
|
||||
vol.All(cv.ensure_list, [vol.In(list(TUBE_LINES))]),
|
||||
})
|
||||
|
||||
|
||||
def setup_platform(hass, config, add_devices, discovery_info=None):
|
||||
"""Set up the Tube sensor."""
|
||||
data = TubeData()
|
||||
data.update()
|
||||
sensors = []
|
||||
for line in config.get(CONF_LINE):
|
||||
sensors.append(LondonTubeSensor(line, data))
|
||||
|
||||
add_devices(sensors, True)
|
||||
|
||||
|
||||
class LondonTubeSensor(Entity):
|
||||
"""Sensor that reads the status of a line from TubeData."""
|
||||
|
||||
ICON = 'mdi:subway'
|
||||
|
||||
def __init__(self, name, data):
|
||||
"""Initialize the sensor."""
|
||||
self._name = name
|
||||
self._data = data
|
||||
self._state = None
|
||||
self._description = None
|
||||
|
||||
@property
|
||||
def name(self):
|
||||
"""Return the name of the sensor."""
|
||||
return self._name
|
||||
|
||||
@property
|
||||
def state(self):
|
||||
"""Return the state of the sensor."""
|
||||
return self._state
|
||||
|
||||
@property
|
||||
def icon(self):
|
||||
"""Icon to use in the frontend, if any."""
|
||||
return self.ICON
|
||||
|
||||
@property
|
||||
def device_state_attributes(self):
|
||||
"""Return other details about the sensor state."""
|
||||
attrs = {}
|
||||
attrs['Description'] = self._description
|
||||
return attrs
|
||||
|
||||
def update(self):
|
||||
"""Update the sensor."""
|
||||
self._data.update()
|
||||
self._state = self._data.data[self.name]['State']
|
||||
self._description = self._data.data[self.name]['Description']
|
||||
|
||||
|
||||
class TubeData(object):
|
||||
"""Get the latest tube data from TFL."""
|
||||
|
||||
def __init__(self):
|
||||
"""Initialize the TubeData object."""
|
||||
self.data = None
|
||||
|
||||
# Update only once in scan interval.
|
||||
@Throttle(SCAN_INTERVAL)
|
||||
def update(self):
|
||||
"""Get the latest data from TFL."""
|
||||
response = requests.get(URL)
|
||||
if response.status_code != 200:
|
||||
_LOGGER.warning("Invalid response from API")
|
||||
else:
|
||||
self.data = parse_api_response(response.json())
|
||||
|
||||
|
||||
def parse_api_response(response):
|
||||
"""Take in the TFL API json response."""
|
||||
lines = [line['name'] for line in response]
|
||||
data_dict = dict.fromkeys(lines)
|
||||
|
||||
for line in response:
|
||||
statuses = [status['statusSeverityDescription']
|
||||
for status in line['lineStatuses']]
|
||||
state = ' + '.join(sorted(set(statuses)))
|
||||
|
||||
if state == 'Good Service':
|
||||
reason = 'Nothing to report'
|
||||
else:
|
||||
reason = ' *** '.join(
|
||||
[status['reason'] for status in line['lineStatuses']])
|
||||
|
||||
attr = {'State': state, 'Description': reason}
|
||||
data_dict[line['name']] = attr
|
||||
|
||||
return data_dict
|
38
tests/components/sensor/test_london_underground.py
Normal file
38
tests/components/sensor/test_london_underground.py
Normal file
@ -0,0 +1,38 @@
|
||||
"""The tests for the tube_state platform."""
|
||||
import unittest
|
||||
import requests_mock
|
||||
|
||||
from homeassistant.components.sensor.london_underground import CONF_LINE, URL
|
||||
from homeassistant.setup import setup_component
|
||||
from tests.common import load_fixture, get_test_home_assistant
|
||||
|
||||
VALID_CONFIG = {
|
||||
'platform': 'london_underground',
|
||||
CONF_LINE: [
|
||||
'London Overground',
|
||||
]
|
||||
}
|
||||
|
||||
|
||||
class TestLondonTubeSensor(unittest.TestCase):
|
||||
"""Test the tube_state platform."""
|
||||
|
||||
def setUp(self):
|
||||
"""Initialize values for this testcase class."""
|
||||
self.hass = get_test_home_assistant()
|
||||
self.config = VALID_CONFIG
|
||||
|
||||
def tearDown(self):
|
||||
"""Stop everything that was started."""
|
||||
self.hass.stop()
|
||||
|
||||
@requests_mock.Mocker()
|
||||
def test_setup(self, mock_req):
|
||||
"""Test for operational tube_state sensor with proper attributes."""
|
||||
mock_req.get(URL, text=load_fixture('london_underground.json'))
|
||||
self.assertTrue(
|
||||
setup_component(self.hass, 'sensor', {'sensor': self.config}))
|
||||
|
||||
state = self.hass.states.get('sensor.london_overground')
|
||||
assert state.state == 'Minor Delays'
|
||||
assert state.attributes.get('Description') == 'something'
|
465
tests/fixtures/london_underground.json
vendored
Normal file
465
tests/fixtures/london_underground.json
vendored
Normal file
@ -0,0 +1,465 @@
|
||||
[
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.Line, Tfl.Api.Presentation.Entities",
|
||||
"id": "bakerloo",
|
||||
"name": "Bakerloo",
|
||||
"modeName": "tube",
|
||||
"disruptions": [],
|
||||
"created": "2017-06-28T11:43:10.703Z",
|
||||
"modified": "2017-06-28T11:43:10.703Z",
|
||||
"lineStatuses": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineStatus, Tfl.Api.Presentation.Entities",
|
||||
"id": 0,
|
||||
"statusSeverity": 10,
|
||||
"statusSeverityDescription": "Good Service",
|
||||
"created": "0001-01-01T00:00:00",
|
||||
"validityPeriods": []
|
||||
}
|
||||
],
|
||||
"routeSections": [],
|
||||
"serviceTypes": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Regular",
|
||||
"uri": "/Line/Route?ids=Bakerloo&serviceTypes=Regular"
|
||||
}
|
||||
],
|
||||
"crowding": {
|
||||
"$type": "Tfl.Api.Presentation.Entities.Crowding, Tfl.Api.Presentation.Entities"
|
||||
}
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.Line, Tfl.Api.Presentation.Entities",
|
||||
"id": "central",
|
||||
"name": "Central",
|
||||
"modeName": "tube",
|
||||
"disruptions": [],
|
||||
"created": "2017-06-28T11:43:10.623Z",
|
||||
"modified": "2017-06-28T11:43:10.623Z",
|
||||
"lineStatuses": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineStatus, Tfl.Api.Presentation.Entities",
|
||||
"id": 0,
|
||||
"statusSeverity": 10,
|
||||
"statusSeverityDescription": "Good Service",
|
||||
"created": "0001-01-01T00:00:00",
|
||||
"validityPeriods": []
|
||||
}
|
||||
],
|
||||
"routeSections": [],
|
||||
"serviceTypes": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Regular",
|
||||
"uri": "/Line/Route?ids=Central&serviceTypes=Regular"
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Night",
|
||||
"uri": "/Line/Route?ids=Central&serviceTypes=Night"
|
||||
}
|
||||
],
|
||||
"crowding": {
|
||||
"$type": "Tfl.Api.Presentation.Entities.Crowding, Tfl.Api.Presentation.Entities"
|
||||
}
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.Line, Tfl.Api.Presentation.Entities",
|
||||
"id": "circle",
|
||||
"name": "Circle",
|
||||
"modeName": "tube",
|
||||
"disruptions": [],
|
||||
"created": "2017-06-28T11:43:10.733Z",
|
||||
"modified": "2017-06-28T11:43:10.733Z",
|
||||
"lineStatuses": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineStatus, Tfl.Api.Presentation.Entities",
|
||||
"id": 0,
|
||||
"statusSeverity": 10,
|
||||
"statusSeverityDescription": "Good Service",
|
||||
"created": "0001-01-01T00:00:00",
|
||||
"validityPeriods": []
|
||||
}
|
||||
],
|
||||
"routeSections": [],
|
||||
"serviceTypes": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Regular",
|
||||
"uri": "/Line/Route?ids=Circle&serviceTypes=Regular"
|
||||
}
|
||||
],
|
||||
"crowding": {
|
||||
"$type": "Tfl.Api.Presentation.Entities.Crowding, Tfl.Api.Presentation.Entities"
|
||||
}
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.Line, Tfl.Api.Presentation.Entities",
|
||||
"id": "district",
|
||||
"name": "District",
|
||||
"modeName": "tube",
|
||||
"disruptions": [],
|
||||
"created": "2017-06-28T11:43:10.623Z",
|
||||
"modified": "2017-06-28T11:43:10.623Z",
|
||||
"lineStatuses": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineStatus, Tfl.Api.Presentation.Entities",
|
||||
"id": 0,
|
||||
"statusSeverity": 10,
|
||||
"statusSeverityDescription": "Good Service",
|
||||
"created": "0001-01-01T00:00:00",
|
||||
"validityPeriods": []
|
||||
}
|
||||
],
|
||||
"routeSections": [],
|
||||
"serviceTypes": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Regular",
|
||||
"uri": "/Line/Route?ids=District&serviceTypes=Regular"
|
||||
}
|
||||
],
|
||||
"crowding": {
|
||||
"$type": "Tfl.Api.Presentation.Entities.Crowding, Tfl.Api.Presentation.Entities"
|
||||
}
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.Line, Tfl.Api.Presentation.Entities",
|
||||
"id": "dlr",
|
||||
"name": "DLR",
|
||||
"modeName": "dlr",
|
||||
"disruptions": [],
|
||||
"created": "2017-06-28T11:43:10.703Z",
|
||||
"modified": "2017-06-28T11:43:10.703Z",
|
||||
"lineStatuses": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineStatus, Tfl.Api.Presentation.Entities",
|
||||
"id": 0,
|
||||
"statusSeverity": 10,
|
||||
"statusSeverityDescription": "Good Service",
|
||||
"created": "0001-01-01T00:00:00",
|
||||
"validityPeriods": []
|
||||
}
|
||||
],
|
||||
"routeSections": [],
|
||||
"serviceTypes": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Regular",
|
||||
"uri": "/Line/Route?ids=DLR&serviceTypes=Regular"
|
||||
}
|
||||
],
|
||||
"crowding": {
|
||||
"$type": "Tfl.Api.Presentation.Entities.Crowding, Tfl.Api.Presentation.Entities"
|
||||
}
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.Line, Tfl.Api.Presentation.Entities",
|
||||
"id": "hammersmith-city",
|
||||
"name": "Hammersmith & City",
|
||||
"modeName": "tube",
|
||||
"disruptions": [],
|
||||
"created": "2017-06-28T11:43:10.67Z",
|
||||
"modified": "2017-06-28T11:43:10.67Z",
|
||||
"lineStatuses": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineStatus, Tfl.Api.Presentation.Entities",
|
||||
"id": 0,
|
||||
"statusSeverity": 10,
|
||||
"statusSeverityDescription": "Good Service",
|
||||
"created": "0001-01-01T00:00:00",
|
||||
"validityPeriods": []
|
||||
}
|
||||
],
|
||||
"routeSections": [],
|
||||
"serviceTypes": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Regular",
|
||||
"uri": "/Line/Route?ids=Hammersmith & City&serviceTypes=Regular"
|
||||
}
|
||||
],
|
||||
"crowding": {
|
||||
"$type": "Tfl.Api.Presentation.Entities.Crowding, Tfl.Api.Presentation.Entities"
|
||||
}
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.Line, Tfl.Api.Presentation.Entities",
|
||||
"id": "jubilee",
|
||||
"name": "Jubilee",
|
||||
"modeName": "tube",
|
||||
"disruptions": [],
|
||||
"created": "2017-06-28T11:43:10.623Z",
|
||||
"modified": "2017-06-28T11:43:10.623Z",
|
||||
"lineStatuses": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineStatus, Tfl.Api.Presentation.Entities",
|
||||
"id": 0,
|
||||
"statusSeverity": 10,
|
||||
"statusSeverityDescription": "Good Service",
|
||||
"created": "0001-01-01T00:00:00",
|
||||
"validityPeriods": []
|
||||
}
|
||||
],
|
||||
"routeSections": [],
|
||||
"serviceTypes": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Regular",
|
||||
"uri": "/Line/Route?ids=Jubilee&serviceTypes=Regular"
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Night",
|
||||
"uri": "/Line/Route?ids=Jubilee&serviceTypes=Night"
|
||||
}
|
||||
],
|
||||
"crowding": {
|
||||
"$type": "Tfl.Api.Presentation.Entities.Crowding, Tfl.Api.Presentation.Entities"
|
||||
}
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.Line, Tfl.Api.Presentation.Entities",
|
||||
"id": "london-overground",
|
||||
"name": "London Overground",
|
||||
"modeName": "overground",
|
||||
"disruptions": [],
|
||||
"created": "2017-06-28T11:43:10.607Z",
|
||||
"modified": "2017-06-28T11:43:10.607Z",
|
||||
"lineStatuses": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineStatus, Tfl.Api.Presentation.Entities",
|
||||
"id": 0,
|
||||
"lineId": "london-overground",
|
||||
"statusSeverity": 9,
|
||||
"statusSeverityDescription": "Minor Delays",
|
||||
"reason": "something",
|
||||
"created": "0001-01-01T00:00:00",
|
||||
"validityPeriods": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.ValidityPeriod, Tfl.Api.Presentation.Entities",
|
||||
"fromDate": "2017-06-29T06:27:21Z",
|
||||
"toDate": "2017-06-30T01:29:00Z",
|
||||
"isNow": true
|
||||
}
|
||||
],
|
||||
"disruption": {
|
||||
"$type": "Tfl.Api.Presentation.Entities.Disruption, Tfl.Api.Presentation.Entities",
|
||||
"category": "RealTime",
|
||||
"categoryDescription": "RealTime",
|
||||
"description": "London Overground: Minor delays Richmond to Stratford and Willesden Junction to Clapham Junction while we fix a faulty train at Richmond, GOOD SERVICE all other routes. ",
|
||||
"affectedRoutes": [],
|
||||
"affectedStops": [],
|
||||
"closureText": "minorDelays"
|
||||
}
|
||||
}
|
||||
],
|
||||
"routeSections": [],
|
||||
"serviceTypes": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Regular",
|
||||
"uri": "/Line/Route?ids=London Overground&serviceTypes=Regular"
|
||||
}
|
||||
],
|
||||
"crowding": {
|
||||
"$type": "Tfl.Api.Presentation.Entities.Crowding, Tfl.Api.Presentation.Entities"
|
||||
}
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.Line, Tfl.Api.Presentation.Entities",
|
||||
"id": "metropolitan",
|
||||
"name": "Metropolitan",
|
||||
"modeName": "tube",
|
||||
"disruptions": [],
|
||||
"created": "2017-06-28T11:43:10.703Z",
|
||||
"modified": "2017-06-28T11:43:10.703Z",
|
||||
"lineStatuses": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineStatus, Tfl.Api.Presentation.Entities",
|
||||
"id": 0,
|
||||
"statusSeverity": 10,
|
||||
"statusSeverityDescription": "Good Service",
|
||||
"created": "0001-01-01T00:00:00",
|
||||
"validityPeriods": []
|
||||
}
|
||||
],
|
||||
"routeSections": [],
|
||||
"serviceTypes": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Regular",
|
||||
"uri": "/Line/Route?ids=Metropolitan&serviceTypes=Regular"
|
||||
}
|
||||
],
|
||||
"crowding": {
|
||||
"$type": "Tfl.Api.Presentation.Entities.Crowding, Tfl.Api.Presentation.Entities"
|
||||
}
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.Line, Tfl.Api.Presentation.Entities",
|
||||
"id": "northern",
|
||||
"name": "Northern",
|
||||
"modeName": "tube",
|
||||
"disruptions": [],
|
||||
"created": "2017-06-28T11:43:10.67Z",
|
||||
"modified": "2017-06-28T11:43:10.67Z",
|
||||
"lineStatuses": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineStatus, Tfl.Api.Presentation.Entities",
|
||||
"id": 0,
|
||||
"statusSeverity": 10,
|
||||
"statusSeverityDescription": "Good Service",
|
||||
"created": "0001-01-01T00:00:00",
|
||||
"validityPeriods": []
|
||||
}
|
||||
],
|
||||
"routeSections": [],
|
||||
"serviceTypes": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Regular",
|
||||
"uri": "/Line/Route?ids=Northern&serviceTypes=Regular"
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Night",
|
||||
"uri": "/Line/Route?ids=Northern&serviceTypes=Night"
|
||||
}
|
||||
],
|
||||
"crowding": {
|
||||
"$type": "Tfl.Api.Presentation.Entities.Crowding, Tfl.Api.Presentation.Entities"
|
||||
}
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.Line, Tfl.Api.Presentation.Entities",
|
||||
"id": "piccadilly",
|
||||
"name": "Piccadilly",
|
||||
"modeName": "tube",
|
||||
"disruptions": [],
|
||||
"created": "2017-06-28T11:43:10.67Z",
|
||||
"modified": "2017-06-28T11:43:10.67Z",
|
||||
"lineStatuses": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineStatus, Tfl.Api.Presentation.Entities",
|
||||
"id": 0,
|
||||
"statusSeverity": 10,
|
||||
"statusSeverityDescription": "Good Service",
|
||||
"created": "0001-01-01T00:00:00",
|
||||
"validityPeriods": []
|
||||
}
|
||||
],
|
||||
"routeSections": [],
|
||||
"serviceTypes": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Regular",
|
||||
"uri": "/Line/Route?ids=Piccadilly&serviceTypes=Regular"
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Night",
|
||||
"uri": "/Line/Route?ids=Piccadilly&serviceTypes=Night"
|
||||
}
|
||||
],
|
||||
"crowding": {
|
||||
"$type": "Tfl.Api.Presentation.Entities.Crowding, Tfl.Api.Presentation.Entities"
|
||||
}
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.Line, Tfl.Api.Presentation.Entities",
|
||||
"id": "tfl-rail",
|
||||
"name": "TfL Rail",
|
||||
"modeName": "tflrail",
|
||||
"disruptions": [],
|
||||
"created": "2017-06-28T11:43:10.657Z",
|
||||
"modified": "2017-06-28T11:43:10.657Z",
|
||||
"lineStatuses": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineStatus, Tfl.Api.Presentation.Entities",
|
||||
"id": 0,
|
||||
"statusSeverity": 10,
|
||||
"statusSeverityDescription": "Good Service",
|
||||
"created": "0001-01-01T00:00:00",
|
||||
"validityPeriods": []
|
||||
}
|
||||
],
|
||||
"routeSections": [],
|
||||
"serviceTypes": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Regular",
|
||||
"uri": "/Line/Route?ids=TfL Rail&serviceTypes=Regular"
|
||||
}
|
||||
],
|
||||
"crowding": {
|
||||
"$type": "Tfl.Api.Presentation.Entities.Crowding, Tfl.Api.Presentation.Entities"
|
||||
}
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.Line, Tfl.Api.Presentation.Entities",
|
||||
"id": "victoria",
|
||||
"name": "Victoria",
|
||||
"modeName": "tube",
|
||||
"disruptions": [],
|
||||
"created": "2017-06-28T11:43:10.607Z",
|
||||
"modified": "2017-06-28T11:43:10.607Z",
|
||||
"lineStatuses": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineStatus, Tfl.Api.Presentation.Entities",
|
||||
"id": 0,
|
||||
"statusSeverity": 10,
|
||||
"statusSeverityDescription": "Good Service",
|
||||
"created": "0001-01-01T00:00:00",
|
||||
"validityPeriods": []
|
||||
}
|
||||
],
|
||||
"routeSections": [],
|
||||
"serviceTypes": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Regular",
|
||||
"uri": "/Line/Route?ids=Victoria&serviceTypes=Regular"
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Night",
|
||||
"uri": "/Line/Route?ids=Victoria&serviceTypes=Night"
|
||||
}
|
||||
],
|
||||
"crowding": {
|
||||
"$type": "Tfl.Api.Presentation.Entities.Crowding, Tfl.Api.Presentation.Entities"
|
||||
}
|
||||
},
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.Line, Tfl.Api.Presentation.Entities",
|
||||
"id": "waterloo-city",
|
||||
"name": "Waterloo & City",
|
||||
"modeName": "tube",
|
||||
"disruptions": [],
|
||||
"created": "2017-06-28T11:43:10.703Z",
|
||||
"modified": "2017-06-28T11:43:10.703Z",
|
||||
"lineStatuses": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineStatus, Tfl.Api.Presentation.Entities",
|
||||
"id": 0,
|
||||
"statusSeverity": 10,
|
||||
"statusSeverityDescription": "Good Service",
|
||||
"created": "0001-01-01T00:00:00",
|
||||
"validityPeriods": []
|
||||
}
|
||||
],
|
||||
"routeSections": [],
|
||||
"serviceTypes": [
|
||||
{
|
||||
"$type": "Tfl.Api.Presentation.Entities.LineServiceTypeInfo, Tfl.Api.Presentation.Entities",
|
||||
"name": "Regular",
|
||||
"uri": "/Line/Route?ids=Waterloo & City&serviceTypes=Regular"
|
||||
}
|
||||
],
|
||||
"crowding": {
|
||||
"$type": "Tfl.Api.Presentation.Entities.Crowding, Tfl.Api.Presentation.Entities"
|
||||
}
|
||||
}
|
||||
]
|
Loading…
x
Reference in New Issue
Block a user