Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

0.50 #8685

Merged
merged 120 commits into from
Jul 29, 2017
Merged

0.50 #8685

Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
120 commits
Select commit Hold shift + click to select a range
5581c62
Fix iFrame panel test
balloob Jul 13, 2017
ba019c7
Make deps directory persistent over upgrades (#7801)
MartinHjelmare Jul 14, 2017
4fde0ff
LIFX: support for multizone (#8399)
amelchio Jul 14, 2017
d8abef9
Add RGB support to switch.flux (#8417)
d0ugal Jul 14, 2017
9373d5e
Fix media_position for cast component (#8452)
dersger Jul 14, 2017
d473f34
Remove km from visibility, add visibility_distance (#8454)
gollo Jul 14, 2017
5829cdf
Radarr sensor fix for issue #8250 (#8456)
hoopty Jul 14, 2017
87b83f3
Accept transition for light.toggle (#8466)
amelchio Jul 14, 2017
6ca828f
Make themes API work even when themes are not defined. (#8473)
andrey-git Jul 14, 2017
543e8bb
Fix check for running inside venv (#8481)
MartinHjelmare Jul 15, 2017
23b65bf
Version bump to 0.50.0dev0
balloob Jul 16, 2017
bffa0d2
Bump to KNXIP 0.5 (#8492)
Jul 16, 2017
d3be056
Expose all components on hass [Concept] (#8490)
balloob Jul 16, 2017
d29bddd
Add bind_hass to components (#8502)
balloob Jul 16, 2017
f6c3832
Fix TP-Link device tracker regression since 0.49 (#8497)
maikelwever Jul 16, 2017
d0275c8
Persistent notification import (#8507)
balloob Jul 16, 2017
ffd3081
Added additional attributes to OwnTracks device_tracker (#8503)
Jul 16, 2017
8c95574
Update dlib_face_detect.py (#8516)
pvizeli Jul 17, 2017
40aafcd
prometheus: Convert fahrenheit to celsius (#8511)
rcloran Jul 17, 2017
f047985
Realfix for dlib (#8517)
pvizeli Jul 17, 2017
cca0d3e
Attach the `chat_id` for a callback query from a chat group (fixes #8…
azogue Jul 17, 2017
1a1571c
Added sensor state rounding (#8499)
fronzbot Jul 17, 2017
c67c20f
fix for a bug introduced with media support in #8282 (#8513)
MikeChristianson Jul 17, 2017
95e0027
Fix KeyError
fabaff Jul 17, 2017
8c9b389
handle timeout errors without logging.exception when updating hue lig…
azogue Jul 17, 2017
b83ff73
Remove deprecated automation keywords (#8510)
amelchio Jul 17, 2017
fde4a7d
Citybikes: Allow None as result for empty slots (#8528)
janLo Jul 17, 2017
dcd6f7a
Return a 0 temperature value when none is found (#8518)
phil-lavin Jul 17, 2017
4ae11c0
Fix #6469 and #6828 (#8537)
titilambert Jul 18, 2017
879c816
Update docstrings (#8536)
fabaff Jul 18, 2017
2aa89cf
Upgrade TwitterAPI to 2.4.6 (#8535)
fabaff Jul 18, 2017
2926621
Fix support for multiple Apple TVs (#8539)
postlund Jul 18, 2017
5e1ff20
Decora: Fix set brightness and improve reconnection (#8522)
titilambert Jul 18, 2017
d54a634
Update demo.py
balloob Jul 18, 2017
1a86fa5
Initial support for Google Wifi/OnHub (#8485)
fronzbot Jul 18, 2017
4ece4bf
Fix exception dlib_face_identify when image is not recognized by face…
tchellomello Jul 19, 2017
6bc0729
[media_extractor] Add support for custom stream queries for media_ext…
minchik Jul 19, 2017
42699b7
Report Harmony remote off if state is unknown (#8547)
jawilson Jul 19, 2017
c8bfcd2
Upgrade the alarmdecoder dependency library from 0.12.1 to 0.12.3. (#…
viswa-swami Jul 19, 2017
c27074e
turn_on_action and turn_off_action with script syntax (#8558)
azogue Jul 19, 2017
f1280d3
Extends Pi-hole sensor to support the new sensors: (#8549)
tchellomello Jul 19, 2017
84ebcd8
Support for Wink Switch and Light groups also fix fan speed selection…
Jul 19, 2017
54755df
Added a service to write to KNX group addressed including documentati…
Jul 20, 2017
9cc3e7e
Handle manual edits to emulated_hue_ids.json (#8560)
jawilson Jul 20, 2017
8a42e15
LIFX: assume default features for unknown products (#8553)
amelchio Jul 20, 2017
ee05a4a
Fix broken status update for lighting4 devices (#8543)
ypollart Jul 20, 2017
8d1999d
Enhance python_script to support "_getitem_" (#8541)
sdague Jul 20, 2017
49c399c
Update persistent deps dir version in config.py (#8479)
balloob Jul 20, 2017
fde291f
Add is_lighting4 to RfxtrxBinarySensor (#8563)
Danielhiversen Jul 20, 2017
bc27d17
Bump pyver to fix exception in fetching colours.
pavoni Jul 20, 2017
966809c
Merge pull request #8564 from home-assistant/bump_pyvera
pavoni Jul 20, 2017
90639d3
Xiaomi gw support (#8555)
Danielhiversen Jul 20, 2017
7189494
fix #8263 corrected Adafruit DHT library version from 1.3.0 to 1.3.2 …
namadori Jul 20, 2017
a5c0831
xiaomi bug fix (#8576)
Danielhiversen Jul 20, 2017
e63a96c
Bumped python-simplisafe version (#8578)
bachya Jul 20, 2017
4f8d2ec
Added Time Remaining and Time Elapsed sensors for octoprint (#8581)
Jul 21, 2017
f6a5e08
upgade xiaomi lib to 0.2 (#8584)
Danielhiversen Jul 21, 2017
fada6d3
Device support for different new sensors of the xiaomi aqara gateway …
syssi Jul 21, 2017
ee15306
Extends Fitbit sensors to track the device battery level (#8583)
tchellomello Jul 21, 2017
4359e0b
xiaomi binary sensor bug fix (#8586)
Danielhiversen Jul 21, 2017
06ceadf
upgrade pywebpush and PyJWT (#8588)
perosb Jul 21, 2017
dc42b63
Support for Wink oauth application authorization (#8208)
Jul 22, 2017
8d31c5f
zha: Update to bellows 0.3.4 (#8594)
rcloran Jul 22, 2017
7bea69c
update frontend
balloob Jul 22, 2017
7edf14e
Add Intent component (#8434)
balloob Jul 22, 2017
8e8ec7a
Remove code in wink.py overwriting hass.data configurator (#8595)
Jul 22, 2017
b4f392b
bump python-mirobo version for more robust protocol handling, make th…
rytilahti Jul 22, 2017
1807b45
Binary sensor ping fixed for hassio (#8573)
gollo Jul 22, 2017
2f08a91
Simplified percent conversion, better logging (#8568)
Jul 22, 2017
9043895
make attributes in the fritzdect module easier to process (#8436)
thomasklingbeil Jul 22, 2017
dac9716
Fix STATION_SCHEMA validation on longitude (#8610)
clkao Jul 23, 2017
c6bf529
Allow set_cover_position in scenes (#8613)
koenekelschot Jul 23, 2017
77d8e39
better but still not great (#8618)
cribbstechnologies Jul 23, 2017
cc2de5e
Upgrade youtube_dl to 2017.7.23 (#8617)
fabaff Jul 23, 2017
486bcc4
Upgrade mypy to 0.520 (#8616)
fabaff Jul 23, 2017
5d810da
REST binary sensor value_template optional (#8596)
philhawthorne Jul 23, 2017
3fec295
Bumped Amcrest version (#8624)
tchellomello Jul 24, 2017
1831a7d
Check if /dev/input/by-id exists (#8601)
schaal Jul 24, 2017
4b449f5
Tado Fix #8606 (#8621)
filcole Jul 24, 2017
f0293ee
prometheus: Fix zwave battery level (#8615)
rcloran Jul 24, 2017
b0b6026
ubus: Make multiple instances work again (#8571)
glance- Jul 24, 2017
f3d9086
Properly slugify switch.flux update service name (#8545)
jawilson Jul 24, 2017
a2abb4a
Update frontend
balloob Jul 24, 2017
654ad41
added onvif camera fix for non-virtual env installations (#8592)
matt2005 Jul 24, 2017
f86bd15
Cleanup old device_tracker stuff (#8627)
pvizeli Jul 24, 2017
1d31137
Update README.rst
balloob Jul 24, 2017
9d9ca64
Update README.rst
balloob Jul 24, 2017
98568b5
Add support for using credstash as a secret store (#8494)
justin8 Jul 24, 2017
ecc1429
Added support for default value when environment variable is missing …
Jul 24, 2017
cc03f7e
Manual alarm with MQTT control (#8257)
colinodell Jul 24, 2017
3b7f16f
Catch and log Lyft API errors (#8635)
emlove Jul 25, 2017
ad7370e
Update frontend
balloob Jul 25, 2017
c2828ba
Tweak conversation/intent/shopping list (#8636)
balloob Jul 25, 2017
cd2703e
Update dependencies cast + discovery (#8646)
balloob Jul 26, 2017
e83816c
Add component Light TPLink (#8643)
gollo Jul 26, 2017
7c12074
Fixes Fitbit sensor to report battery level with the expected device …
tchellomello Jul 26, 2017
81a27e7
Upgrade aiolifx (#8648)
amelchio Jul 26, 2017
fff269e
Velbus (#8076)
thomasdelaet Jul 26, 2017
abcfcdd
Yahoo Weather update, supports forecast for more days (#8626)
fanthos Jul 26, 2017
438edc5
History performance improvements for single-entity requests (#8632)
OverloadUT Jul 26, 2017
3318f02
Add transition support to light.zha (#8548)
jawilson Jul 26, 2017
3b4ea86
Add uk_transport component. (#8600)
robmarkcole Jul 26, 2017
f5eeb25
Added support for SerenaHoneycombShades to Lutron Caseta (#8662)
kfcook Jul 27, 2017
9d5c61b
MQTT Switch: add availability_topic for online/offline status (#8593)
abmantis Jul 27, 2017
1e8c00a
Adding support for mapping keys to value in statsd (#8665)
Khabi Jul 27, 2017
b59c299
Upgrade fuzzywuzzy to 0.15.1 (#8671)
fabaff Jul 27, 2017
4fcaea2
Upgrade libnacl to 1.5.2 (#8670)
fabaff Jul 27, 2017
74581b5
Upgrade sqlalchemy to 1.1.12 (#8669)
fabaff Jul 27, 2017
9e6817b
Upgrade pyhomematic to 0.1.30 (#8673)
danielperna84 Jul 27, 2017
51108b8
Hass.io: logo support / timeout handling (#8668)
pvizeli Jul 27, 2017
0ab0e35
Updated pysnmp to 4.3.9 (#8675)
bgehrich Jul 27, 2017
e8ce418
Fix COMMAND_CLASS_BARRIER_OPERATOR for dev branch of OpenZwave (#8574)
firstof9 Jul 27, 2017
0c97fe7
Version bump to 0.50
Jul 28, 2017
0a6d519
Merge remote-tracking branch 'origin/master' into release-0-50
Jul 28, 2017
828c469
Fix Lint
balloob Jul 29, 2017
56f4486
Update frontend
balloob Jul 29, 2017
f0e5f68
Shopping List: edit name / complete status (#8666)
balloob Jul 29, 2017
c376bc2
Support for Wink local control (#8607)
Jul 28, 2017
12dec93
Update frontend
balloob Jul 29, 2017
a760673
Persist shopping list + clear completed (#8697)
balloob Jul 29, 2017
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Add uk_transport component. (#8600)
  • Loading branch information
robmarkcole authored and lwis committed Jul 26, 2017
commit 3b4ea864a10c6923bec25e643d6d27ba022a7832
275 changes: 275 additions & 0 deletions homeassistant/components/sensor/uk_transport.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,275 @@
"""Support for UK public transport data provided by transportapi.com.

For more details about this platform, please refer to the documentation at
https://home-assistant.io/components/sensor.uk_transport/
"""
import logging
import re
from datetime import datetime, timedelta
import requests
import voluptuous as vol

from homeassistant.components.sensor import PLATFORM_SCHEMA
from homeassistant.helpers.entity import Entity
from homeassistant.util import Throttle
import homeassistant.helpers.config_validation as cv

_LOGGER = logging.getLogger(__name__)

ATTR_ATCOCODE = 'atcocode'
ATTR_LOCALITY = 'locality'
ATTR_STOP_NAME = 'stop_name'
ATTR_REQUEST_TIME = 'request_time'
ATTR_NEXT_BUSES = 'next_buses'
ATTR_STATION_CODE = 'station_code'
ATTR_CALLING_AT = 'calling_at'
ATTR_NEXT_TRAINS = 'next_trains'

CONF_API_APP_KEY = 'app_key'
CONF_API_APP_ID = 'app_id'
CONF_QUERIES = 'queries'
CONF_MODE = 'mode'
CONF_ORIGIN = 'origin'
CONF_DESTINATION = 'destination'

_QUERY_SCHEME = vol.Schema({
vol.Required(CONF_MODE):
vol.All(cv.ensure_list, [vol.In(list(['bus', 'train']))]),
vol.Required(CONF_ORIGIN): cv.string,
vol.Required(CONF_DESTINATION): cv.string,
})

PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
vol.Required(CONF_API_APP_ID): cv.string,
vol.Required(CONF_API_APP_KEY): cv.string,
vol.Required(CONF_QUERIES): [_QUERY_SCHEME],
})


def setup_platform(hass, config, add_devices, discovery_info=None):
"""Get the uk_transport sensor."""
sensors = []
number_sensors = len(config.get(CONF_QUERIES))
interval = timedelta(seconds=87*number_sensors)

for query in config.get(CONF_QUERIES):
if 'bus' in query.get(CONF_MODE):
stop_atcocode = query.get(CONF_ORIGIN)
bus_direction = query.get(CONF_DESTINATION)
sensors.append(
UkTransportLiveBusTimeSensor(
config.get(CONF_API_APP_ID),
config.get(CONF_API_APP_KEY),
stop_atcocode,
bus_direction,
interval))

elif 'train' in query.get(CONF_MODE):
station_code = query.get(CONF_ORIGIN)
calling_at = query.get(CONF_DESTINATION)
sensors.append(
UkTransportLiveTrainTimeSensor(
config.get(CONF_API_APP_ID),
config.get(CONF_API_APP_KEY),
station_code,
calling_at,
interval))

add_devices(sensors, True)


class UkTransportSensor(Entity):
"""
Sensor that reads the UK transport web API.

transportapi.com provides comprehensive transport data for UK train, tube
and bus travel across the UK via simple JSON API. Subclasses of this
base class can be used to access specific types of information.
"""

TRANSPORT_API_URL_BASE = "https://transportapi.com/v3/uk/"
ICON = 'mdi:train'

def __init__(self, name, api_app_id, api_app_key, url):
"""Initialize the sensor."""
self._data = {}
self._api_app_id = api_app_id
self._api_app_key = api_app_key
self._url = self.TRANSPORT_API_URL_BASE + url
self._name = name
self._state = 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 unit_of_measurement(self):
"""Return the unit this state is expressed in."""
return "min"

@property
def icon(self):
"""Icon to use in the frontend, if any."""
return self.ICON

def _do_api_request(self, params):
"""Perform an API request."""
request_params = dict({
'app_id': self._api_app_id,
'app_key': self._api_app_key,
}, **params)

response = requests.get(self._url, params=request_params)
if response.status_code != 200:
_LOGGER.warning('Invalid response from API')
elif 'error' in response.json():
if 'exceeded' in response.json()['error']:
self._state = 'Useage limites exceeded'
if 'invalid' in response.json()['error']:
self._state = 'Credentials invalid'
else:
self._data = response.json()


class UkTransportLiveBusTimeSensor(UkTransportSensor):
"""Live bus time sensor from UK transportapi.com."""

ICON = 'mdi:bus'

def __init__(self, api_app_id, api_app_key,
stop_atcocode, bus_direction, interval):
"""Construct a live bus time sensor."""
self._stop_atcocode = stop_atcocode
self._bus_direction = bus_direction
self._next_buses = []
self._destination_re = re.compile(
'{}'.format(bus_direction), re.IGNORECASE
)

sensor_name = 'Next bus to {}'.format(bus_direction)
stop_url = 'bus/stop/{}/live.json'.format(stop_atcocode)

UkTransportSensor.__init__(
self, sensor_name, api_app_id, api_app_key, stop_url
)
self.update = Throttle(interval)(self._update)

def _update(self):
"""Get the latest live departure data for the specified stop."""
params = {'group': 'route', 'nextbuses': 'no'}

self._do_api_request(params)

if self._data != {}:
self._next_buses = []

for (route, departures) in self._data['departures'].items():
for departure in departures:
if self._destination_re.search(departure['direction']):
self._next_buses.append({
'route': route,
'direction': departure['direction'],
'scheduled': departure['aimed_departure_time'],
'estimated': departure['best_departure_estimate']
})

self._state = min(map(
_delta_mins, [bus['scheduled'] for bus in self._next_buses]
))

@property
def device_state_attributes(self):
"""Return other details about the sensor state."""
attrs = {}
if self._data is not None:
for key in [
ATTR_ATCOCODE, ATTR_LOCALITY, ATTR_STOP_NAME,
ATTR_REQUEST_TIME
]:
attrs[key] = self._data.get(key)
attrs[ATTR_NEXT_BUSES] = self._next_buses
return attrs


class UkTransportLiveTrainTimeSensor(UkTransportSensor):
"""Live train time sensor from UK transportapi.com."""

ICON = 'mdi:train'

def __init__(self, api_app_id, api_app_key,
station_code, calling_at, interval):
"""Construct a live bus time sensor."""
self._station_code = station_code
self._calling_at = calling_at
self._next_trains = []

sensor_name = 'Next train to {}'.format(calling_at)
query_url = 'train/station/{}/live.json'.format(station_code)

UkTransportSensor.__init__(
self, sensor_name, api_app_id, api_app_key, query_url
)
self.update = Throttle(interval)(self._update)

def _update(self):
"""Get the latest live departure data for the specified stop."""
params = {'darwin': 'false',
'calling_at': self._calling_at,
'train_status': 'passenger'}

self._do_api_request(params)
self._next_trains = []

if self._data != {}:
if self._data['departures']['all'] == []:
self._state = 'No departures'
else:
for departure in self._data['departures']['all']:
self._next_trains.append({
'origin_name': departure['origin_name'],
'destination_name': departure['destination_name'],
'status': departure['status'],
'scheduled': departure['aimed_departure_time'],
'estimated': departure['expected_departure_time'],
'platform': departure['platform'],
'operator_name': departure['operator_name']
})

self._state = min(map(
_delta_mins,
[train['scheduled'] for train in self._next_trains]
))

@property
def device_state_attributes(self):
"""Return other details about the sensor state."""
attrs = {}
if self._data is not None:
attrs[ATTR_STATION_CODE] = self._station_code
attrs[ATTR_CALLING_AT] = self._calling_at
if self._next_trains:
attrs[ATTR_NEXT_TRAINS] = self._next_trains
return attrs


def _delta_mins(hhmm_time_str):
"""Calculate time delta in minutes to a time in hh:mm format."""
now = datetime.now()
hhmm_time = datetime.strptime(hhmm_time_str, '%H:%M')

hhmm_datetime = datetime(
now.year, now.month, now.day,
hour=hhmm_time.hour, minute=hhmm_time.minute
)
if hhmm_datetime < now:
hhmm_datetime += timedelta(days=1)

delta_mins = (hhmm_datetime - now).seconds // 60
return delta_mins
93 changes: 93 additions & 0 deletions tests/components/sensor/test_uk_transport.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,93 @@
"""The tests for the uk_transport platform."""
import re

import requests_mock
import unittest

from homeassistant.components.sensor.uk_transport import (
UkTransportSensor,
ATTR_ATCOCODE, ATTR_LOCALITY, ATTR_STOP_NAME, ATTR_NEXT_BUSES,
ATTR_STATION_CODE, ATTR_CALLING_AT, ATTR_NEXT_TRAINS,
CONF_API_APP_KEY, CONF_API_APP_ID)
from homeassistant.setup import setup_component
from tests.common import load_fixture, get_test_home_assistant

BUS_ATCOCODE = '340000368SHE'
BUS_DIRECTION = 'Wantage'
TRAIN_STATION_CODE = 'WIM'
TRAIN_DESTINATION_NAME = 'WAT'

VALID_CONFIG = {
'platform': 'uk_transport',
CONF_API_APP_ID: 'foo',
CONF_API_APP_KEY: 'ebcd1234',
'queries': [{
'mode': 'bus',
'origin': BUS_ATCOCODE,
'destination': BUS_DIRECTION},
{
'mode': 'train',
'origin': TRAIN_STATION_CODE,
'destination': TRAIN_DESTINATION_NAME}]
}


class TestUkTransportSensor(unittest.TestCase):
"""Test the uk_transport 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_bus(self, mock_req):
"""Test for operational uk_transport sensor with proper attributes."""
with requests_mock.Mocker() as mock_req:
uri = re.compile(UkTransportSensor.TRANSPORT_API_URL_BASE + '*')
mock_req.get(uri, text=load_fixture('uk_transport_bus.json'))
self.assertTrue(
setup_component(self.hass, 'sensor', {'sensor': self.config}))

bus_state = self.hass.states.get('sensor.next_bus_to_wantage')

assert type(bus_state.state) == str
assert bus_state.name == 'Next bus to {}'.format(BUS_DIRECTION)
assert bus_state.attributes.get(ATTR_ATCOCODE) == BUS_ATCOCODE
assert bus_state.attributes.get(ATTR_LOCALITY) == 'Harwell Campus'
assert bus_state.attributes.get(ATTR_STOP_NAME) == 'Bus Station'
assert len(bus_state.attributes.get(ATTR_NEXT_BUSES)) == 2

direction_re = re.compile(BUS_DIRECTION)
for bus in bus_state.attributes.get(ATTR_NEXT_BUSES):
print(bus['direction'], direction_re.match(bus['direction']))
assert direction_re.search(bus['direction']) is not None

@requests_mock.Mocker()
def test_train(self, mock_req):
"""Test for operational uk_transport sensor with proper attributes."""
with requests_mock.Mocker() as mock_req:
uri = re.compile(UkTransportSensor.TRANSPORT_API_URL_BASE + '*')
mock_req.get(uri, text=load_fixture('uk_transport_train.json'))
self.assertTrue(
setup_component(self.hass, 'sensor', {'sensor': self.config}))

train_state = self.hass.states.get('sensor.next_train_to_WAT')

assert type(train_state.state) == str
assert train_state.name == 'Next train to {}'.format(
TRAIN_DESTINATION_NAME)
assert train_state.attributes.get(
ATTR_STATION_CODE) == TRAIN_STATION_CODE
assert train_state.attributes.get(
ATTR_CALLING_AT) == TRAIN_DESTINATION_NAME
assert len(train_state.attributes.get(ATTR_NEXT_TRAINS)) == 25

assert train_state.attributes.get(
ATTR_NEXT_TRAINS)[0]['destination_name'] == 'London Waterloo'
assert train_state.attributes.get(
ATTR_NEXT_TRAINS)[0]['estimated'] == '06:13'
Loading