Extension sensor alpha_vantage (#11427)
* Extended sensor alpha_vantage * Improved error handling if symbol is not found. Now we add the symbols that were found, * Added option to give names and currencies to symbols. * Added support to read foreign exchange information. * Icons are selected based on the currency (where MDI has a matching icon) * added missing line at the end of the file... * renamed variable as required by pylint * Fix typos, ordering, and stylepull/11531/head
parent
46ad194097
commit
dff36b8087
|
@ -10,7 +10,8 @@ import logging
|
||||||
import voluptuous as vol
|
import voluptuous as vol
|
||||||
|
|
||||||
from homeassistant.components.sensor import PLATFORM_SCHEMA
|
from homeassistant.components.sensor import PLATFORM_SCHEMA
|
||||||
from homeassistant.const import ATTR_ATTRIBUTION, CONF_API_KEY
|
from homeassistant.const import (
|
||||||
|
ATTR_ATTRIBUTION, CONF_API_KEY, CONF_CURRENCY, CONF_NAME)
|
||||||
import homeassistant.helpers.config_validation as cv
|
import homeassistant.helpers.config_validation as cv
|
||||||
from homeassistant.helpers.entity import Entity
|
from homeassistant.helpers.entity import Entity
|
||||||
|
|
||||||
|
@ -23,25 +24,62 @@ ATTR_HIGH = 'high'
|
||||||
ATTR_LOW = 'low'
|
ATTR_LOW = 'low'
|
||||||
ATTR_VOLUME = 'volume'
|
ATTR_VOLUME = 'volume'
|
||||||
|
|
||||||
CONF_ATTRIBUTION = "Stock market information provided by Alpha Vantage."
|
CONF_ATTRIBUTION = "Stock market information provided by Alpha Vantage"
|
||||||
|
CONF_FOREIGN_EXCHANGE = 'foreign_exchange'
|
||||||
|
CONF_FROM = 'from'
|
||||||
|
CONF_SYMBOL = 'symbol'
|
||||||
CONF_SYMBOLS = 'symbols'
|
CONF_SYMBOLS = 'symbols'
|
||||||
|
CONF_TO = 'to'
|
||||||
|
|
||||||
DEFAULT_SYMBOL = 'GOOGL'
|
DEFAULT_SYMBOL = {
|
||||||
|
CONF_CURRENCY: 'USD',
|
||||||
|
CONF_NAME: 'Google',
|
||||||
|
CONF_SYMBOL: 'GOOGL',
|
||||||
|
}
|
||||||
|
|
||||||
ICON = 'mdi:currency-usd'
|
DEFAULT_CURRENCY = {
|
||||||
|
CONF_FROM: 'BTC',
|
||||||
|
CONF_NAME: 'Bitcon',
|
||||||
|
CONF_TO: 'USD',
|
||||||
|
}
|
||||||
|
|
||||||
|
ICONS = {
|
||||||
|
'BTC': 'mdi:currency-btc',
|
||||||
|
'EUR': 'mdi:currency-eur',
|
||||||
|
'GBP': 'mdi:currency-gbp',
|
||||||
|
'INR': 'mdi:currency-inr',
|
||||||
|
'RUB': 'mdi:currency-rub',
|
||||||
|
'TRY': 'mdi: currency-try',
|
||||||
|
'USD': 'mdi:currency-usd',
|
||||||
|
}
|
||||||
|
|
||||||
SCAN_INTERVAL = timedelta(minutes=5)
|
SCAN_INTERVAL = timedelta(minutes=5)
|
||||||
|
|
||||||
|
SYMBOL_SCHEMA = vol.Schema({
|
||||||
|
vol.Required(CONF_SYMBOL): cv.string,
|
||||||
|
vol.Optional(CONF_CURRENCY): cv.string,
|
||||||
|
vol.Optional(CONF_NAME): cv.string,
|
||||||
|
})
|
||||||
|
|
||||||
|
CURRENCY_SCHEMA = vol.Schema({
|
||||||
|
vol.Required(CONF_FROM): cv.string,
|
||||||
|
vol.Required(CONF_TO): cv.string,
|
||||||
|
vol.Optional(CONF_NAME): cv.string,
|
||||||
|
})
|
||||||
|
|
||||||
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
||||||
vol.Required(CONF_API_KEY): cv.string,
|
vol.Required(CONF_API_KEY): cv.string,
|
||||||
|
vol.Optional(CONF_FOREIGN_EXCHANGE, default=[DEFAULT_CURRENCY]):
|
||||||
|
vol.All(cv.ensure_list, [CURRENCY_SCHEMA]),
|
||||||
vol.Optional(CONF_SYMBOLS, default=[DEFAULT_SYMBOL]):
|
vol.Optional(CONF_SYMBOLS, default=[DEFAULT_SYMBOL]):
|
||||||
vol.All(cv.ensure_list, [cv.string]),
|
vol.All(cv.ensure_list, [SYMBOL_SCHEMA]),
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
||||||
def setup_platform(hass, config, add_devices, discovery_info=None):
|
def setup_platform(hass, config, add_devices, discovery_info=None):
|
||||||
"""Set up the Alpha Vantage sensor."""
|
"""Set up the Alpha Vantage sensor."""
|
||||||
from alpha_vantage.timeseries import TimeSeries
|
from alpha_vantage.timeseries import TimeSeries
|
||||||
|
from alpha_vantage.foreignexchange import ForeignExchange
|
||||||
|
|
||||||
api_key = config.get(CONF_API_KEY)
|
api_key = config.get(CONF_API_KEY)
|
||||||
symbols = config.get(CONF_SYMBOLS)
|
symbols = config.get(CONF_SYMBOLS)
|
||||||
|
@ -51,13 +89,26 @@ def setup_platform(hass, config, add_devices, discovery_info=None):
|
||||||
dev = []
|
dev = []
|
||||||
for symbol in symbols:
|
for symbol in symbols:
|
||||||
try:
|
try:
|
||||||
timeseries.get_intraday(symbol)
|
timeseries.get_intraday(symbol[CONF_SYMBOL])
|
||||||
except ValueError:
|
except ValueError:
|
||||||
_LOGGER.error(
|
_LOGGER.error(
|
||||||
"API Key is not valid or symbol '%s' not known", symbol)
|
"API Key is not valid or symbol '%s' not known", symbol)
|
||||||
return
|
|
||||||
dev.append(AlphaVantageSensor(timeseries, symbol))
|
dev.append(AlphaVantageSensor(timeseries, symbol))
|
||||||
|
|
||||||
|
forex = ForeignExchange(key=api_key)
|
||||||
|
for conversion in config.get(CONF_FOREIGN_EXCHANGE):
|
||||||
|
from_cur = conversion.get(CONF_FROM)
|
||||||
|
to_cur = conversion.get(CONF_TO)
|
||||||
|
try:
|
||||||
|
forex.get_currency_exchange_rate(
|
||||||
|
from_currency=from_cur, to_currency=to_cur)
|
||||||
|
except ValueError as error:
|
||||||
|
_LOGGER.error(
|
||||||
|
"API Key is not valid or currencies '%s'/'%s' not known",
|
||||||
|
from_cur, to_cur)
|
||||||
|
_LOGGER.debug(str(error))
|
||||||
|
dev.append(AlphaVantageForeignExchange(forex, conversion))
|
||||||
|
|
||||||
add_devices(dev, True)
|
add_devices(dev, True)
|
||||||
|
|
||||||
|
|
||||||
|
@ -66,11 +117,12 @@ class AlphaVantageSensor(Entity):
|
||||||
|
|
||||||
def __init__(self, timeseries, symbol):
|
def __init__(self, timeseries, symbol):
|
||||||
"""Initialize the sensor."""
|
"""Initialize the sensor."""
|
||||||
self._name = symbol
|
self._symbol = symbol[CONF_SYMBOL]
|
||||||
|
self._name = symbol.get(CONF_NAME, self._symbol)
|
||||||
self._timeseries = timeseries
|
self._timeseries = timeseries
|
||||||
self._symbol = symbol
|
|
||||||
self.values = None
|
self.values = None
|
||||||
self._unit_of_measurement = None
|
self._unit_of_measurement = symbol.get(CONF_CURRENCY, self._symbol)
|
||||||
|
self._icon = ICONS.get(symbol.get(CONF_CURRENCY, 'USD'))
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def name(self):
|
def name(self):
|
||||||
|
@ -80,7 +132,7 @@ class AlphaVantageSensor(Entity):
|
||||||
@property
|
@property
|
||||||
def unit_of_measurement(self):
|
def unit_of_measurement(self):
|
||||||
"""Return the unit of measurement of this entity, if any."""
|
"""Return the unit of measurement of this entity, if any."""
|
||||||
return self._symbol
|
return self._unit_of_measurement
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def state(self):
|
def state(self):
|
||||||
|
@ -102,9 +154,61 @@ class AlphaVantageSensor(Entity):
|
||||||
@property
|
@property
|
||||||
def icon(self):
|
def icon(self):
|
||||||
"""Return the icon to use in the frontend, if any."""
|
"""Return the icon to use in the frontend, if any."""
|
||||||
return ICON
|
return self._icon
|
||||||
|
|
||||||
def update(self):
|
def update(self):
|
||||||
"""Get the latest data and updates the states."""
|
"""Get the latest data and updates the states."""
|
||||||
all_values, _ = self._timeseries.get_intraday(self._symbol)
|
all_values, _ = self._timeseries.get_intraday(self._symbol)
|
||||||
self.values = next(iter(all_values.values()))
|
self.values = next(iter(all_values.values()))
|
||||||
|
|
||||||
|
|
||||||
|
class AlphaVantageForeignExchange(Entity):
|
||||||
|
"""Sensor for foreign exchange rates."""
|
||||||
|
|
||||||
|
def __init__(self, foreign_exchange, config):
|
||||||
|
"""Initialize the sensor."""
|
||||||
|
self._foreign_exchange = foreign_exchange
|
||||||
|
self._from_currency = config.get(CONF_FROM)
|
||||||
|
self._to_currency = config.get(CONF_TO)
|
||||||
|
if CONF_NAME in config:
|
||||||
|
self._name = config.get(CONF_NAME)
|
||||||
|
else:
|
||||||
|
self._name = '{}/{}'.format(self._to_currency, self._from_currency)
|
||||||
|
self._unit_of_measurement = self._to_currency
|
||||||
|
self._icon = ICONS.get(self._from_currency, 'USD')
|
||||||
|
self.values = None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self):
|
||||||
|
"""Return the name of the sensor."""
|
||||||
|
return self._name
|
||||||
|
|
||||||
|
@property
|
||||||
|
def unit_of_measurement(self):
|
||||||
|
"""Return the unit of measurement of this entity, if any."""
|
||||||
|
return self._unit_of_measurement
|
||||||
|
|
||||||
|
@property
|
||||||
|
def state(self):
|
||||||
|
"""Return the state of the sensor."""
|
||||||
|
return self.values['5. Exchange Rate']
|
||||||
|
|
||||||
|
@property
|
||||||
|
def icon(self):
|
||||||
|
"""Return the icon to use in the frontend, if any."""
|
||||||
|
return self._icon
|
||||||
|
|
||||||
|
@property
|
||||||
|
def device_state_attributes(self):
|
||||||
|
"""Return the state attributes."""
|
||||||
|
if self.values is not None:
|
||||||
|
return {
|
||||||
|
ATTR_ATTRIBUTION: CONF_ATTRIBUTION,
|
||||||
|
CONF_FROM: self._from_currency,
|
||||||
|
CONF_TO: self._to_currency,
|
||||||
|
}
|
||||||
|
|
||||||
|
def update(self):
|
||||||
|
"""Get the latest data and updates the states."""
|
||||||
|
self.values, _ = self._foreign_exchange.get_currency_exchange_rate(
|
||||||
|
from_currency=self._from_currency, to_currency=self._to_currency)
|
||||||
|
|
Loading…
Reference in New Issue