2015-02-01 04:06:30 +00:00
|
|
|
"""
|
|
|
|
Provide pre-made queries on top of the recorder component.
|
2015-10-25 14:04:37 +00:00
|
|
|
|
|
|
|
For more details about this component, please refer to the documentation at
|
2015-11-09 12:12:18 +00:00
|
|
|
https://home-assistant.io/components/history/
|
2015-02-01 04:06:30 +00:00
|
|
|
"""
|
2015-01-31 18:31:16 +00:00
|
|
|
import re
|
2016-02-19 05:27:50 +00:00
|
|
|
from collections import defaultdict
|
2015-04-29 02:12:05 +00:00
|
|
|
from datetime import timedelta
|
2015-02-02 02:00:30 +00:00
|
|
|
from itertools import groupby
|
2015-01-31 18:31:16 +00:00
|
|
|
|
2016-03-05 18:28:48 +00:00
|
|
|
from homeassistant.components import recorder, script
|
2016-02-19 05:27:50 +00:00
|
|
|
import homeassistant.util.dt as dt_util
|
2015-06-16 05:40:57 +00:00
|
|
|
from homeassistant.const import HTTP_BAD_REQUEST
|
2015-01-31 18:31:16 +00:00
|
|
|
|
|
|
|
DOMAIN = 'history'
|
|
|
|
DEPENDENCIES = ['recorder', 'http']
|
|
|
|
|
2016-01-23 20:36:43 +00:00
|
|
|
SIGNIFICANT_DOMAINS = ('thermostat',)
|
2016-03-05 17:49:04 +00:00
|
|
|
IGNORE_DOMAINS = ('zone', 'scene',)
|
2016-01-23 20:36:43 +00:00
|
|
|
|
2015-06-16 05:40:57 +00:00
|
|
|
URL_HISTORY_PERIOD = re.compile(
|
|
|
|
r'/api/history/period(?:/(?P<date>\d{4}-\d{1,2}-\d{1,2})|)')
|
|
|
|
|
2015-01-31 18:31:16 +00:00
|
|
|
|
|
|
|
def last_5_states(entity_id):
|
2016-03-07 17:49:31 +00:00
|
|
|
"""Return the last 5 states for entity_id."""
|
2015-02-02 02:00:30 +00:00
|
|
|
entity_id = entity_id.lower()
|
|
|
|
|
2015-02-01 04:06:30 +00:00
|
|
|
query = """
|
|
|
|
SELECT * FROM states WHERE entity_id=? AND
|
2015-02-02 02:00:30 +00:00
|
|
|
last_changed=last_updated
|
2015-05-01 04:03:01 +00:00
|
|
|
ORDER BY state_id DESC LIMIT 0, 5
|
2015-02-02 02:00:30 +00:00
|
|
|
"""
|
2015-01-31 18:31:16 +00:00
|
|
|
|
2015-02-01 04:06:30 +00:00
|
|
|
return recorder.query_states(query, (entity_id, ))
|
2015-01-31 18:31:16 +00:00
|
|
|
|
|
|
|
|
2016-01-23 20:36:43 +00:00
|
|
|
def get_significant_states(start_time, end_time=None, entity_id=None):
|
2016-03-07 17:49:31 +00:00
|
|
|
"""
|
|
|
|
Return states changes during UTC period start_time - end_time.
|
2016-01-23 20:36:43 +00:00
|
|
|
|
|
|
|
Significant states are all states where there is a state change,
|
|
|
|
as well as all states from certain domains (for instance
|
|
|
|
thermostat so that we get current temperature in our graphs).
|
|
|
|
"""
|
|
|
|
where = """
|
2016-03-05 17:49:04 +00:00
|
|
|
(domain IN ({}) OR last_changed=last_updated)
|
|
|
|
AND domain NOT IN ({}) AND last_updated > ?
|
|
|
|
""".format(",".join("'%s'" % x for x in SIGNIFICANT_DOMAINS),
|
|
|
|
",".join("'%s'" % x for x in IGNORE_DOMAINS))
|
2016-01-23 20:36:43 +00:00
|
|
|
|
|
|
|
data = [start_time]
|
|
|
|
|
|
|
|
if end_time is not None:
|
|
|
|
where += "AND last_updated < ? "
|
|
|
|
data.append(end_time)
|
|
|
|
|
|
|
|
if entity_id is not None:
|
|
|
|
where += "AND entity_id = ? "
|
|
|
|
data.append(entity_id.lower())
|
|
|
|
|
|
|
|
query = ("SELECT * FROM states WHERE {} "
|
|
|
|
"ORDER BY entity_id, last_updated ASC").format(where)
|
|
|
|
|
2016-03-05 18:28:48 +00:00
|
|
|
states = (state for state in recorder.query_states(query, data)
|
|
|
|
if _is_significant(state))
|
2016-01-23 20:36:43 +00:00
|
|
|
|
|
|
|
return states_to_json(states, start_time, entity_id)
|
|
|
|
|
|
|
|
|
2015-02-02 02:00:30 +00:00
|
|
|
def state_changes_during_period(start_time, end_time=None, entity_id=None):
|
2016-03-08 16:55:57 +00:00
|
|
|
"""Return states changes during UTC period start_time - end_time."""
|
2015-02-02 02:00:30 +00:00
|
|
|
where = "last_changed=last_updated AND last_changed > ? "
|
|
|
|
data = [start_time]
|
|
|
|
|
|
|
|
if end_time is not None:
|
|
|
|
where += "AND last_changed < ? "
|
|
|
|
data.append(end_time)
|
|
|
|
|
|
|
|
if entity_id is not None:
|
|
|
|
where += "AND entity_id = ? "
|
|
|
|
data.append(entity_id.lower())
|
|
|
|
|
|
|
|
query = ("SELECT * FROM states WHERE {} "
|
|
|
|
"ORDER BY entity_id, last_changed ASC").format(where)
|
|
|
|
|
|
|
|
states = recorder.query_states(query, data)
|
|
|
|
|
2016-01-23 20:36:43 +00:00
|
|
|
return states_to_json(states, start_time, entity_id)
|
2015-02-06 06:53:36 +00:00
|
|
|
|
|
|
|
|
2015-04-29 02:12:05 +00:00
|
|
|
def get_states(utc_point_in_time, entity_ids=None, run=None):
|
2016-03-08 16:55:57 +00:00
|
|
|
"""Return the states at a specific point in time."""
|
2015-02-07 21:23:01 +00:00
|
|
|
if run is None:
|
2015-04-29 02:12:05 +00:00
|
|
|
run = recorder.run_information(utc_point_in_time)
|
2015-02-06 06:53:36 +00:00
|
|
|
|
2015-04-29 02:12:05 +00:00
|
|
|
# History did not run before utc_point_in_time
|
2015-03-29 16:42:24 +00:00
|
|
|
if run is None:
|
|
|
|
return []
|
|
|
|
|
2015-02-07 21:23:01 +00:00
|
|
|
where = run.where_after_start_run + "AND created < ? "
|
2015-04-29 02:12:05 +00:00
|
|
|
where_data = [utc_point_in_time]
|
2015-02-07 21:23:01 +00:00
|
|
|
|
|
|
|
if entity_ids is not None:
|
|
|
|
where += "AND entity_id IN ({}) ".format(
|
|
|
|
",".join(['?'] * len(entity_ids)))
|
|
|
|
where_data.extend(entity_ids)
|
|
|
|
|
|
|
|
query = """
|
|
|
|
SELECT * FROM states
|
|
|
|
INNER JOIN (
|
|
|
|
SELECT max(state_id) AS max_state_id
|
|
|
|
FROM states WHERE {}
|
|
|
|
GROUP BY entity_id)
|
|
|
|
WHERE state_id = max_state_id
|
|
|
|
""".format(where)
|
|
|
|
|
|
|
|
return recorder.query_states(query, where_data)
|
|
|
|
|
|
|
|
|
2016-01-23 20:36:43 +00:00
|
|
|
def states_to_json(states, start_time, entity_id):
|
2016-03-08 16:55:57 +00:00
|
|
|
"""Convert SQL results into JSON friendly data structure.
|
2016-01-23 20:36:43 +00:00
|
|
|
|
|
|
|
This takes our state list and turns it into a JSON friendly data
|
|
|
|
structure {'entity_id': [list of states], 'entity_id2': [list of states]}
|
|
|
|
|
|
|
|
We also need to go back and create a synthetic zero data point for
|
|
|
|
each list of states, otherwise our graphs won't start on the Y
|
|
|
|
axis correctly.
|
|
|
|
"""
|
|
|
|
result = defaultdict(list)
|
|
|
|
|
|
|
|
entity_ids = [entity_id] if entity_id is not None else None
|
|
|
|
|
|
|
|
# Get the states at the start time
|
|
|
|
for state in get_states(start_time, entity_ids):
|
|
|
|
state.last_changed = start_time
|
|
|
|
state.last_updated = start_time
|
|
|
|
result[state.entity_id].append(state)
|
|
|
|
|
|
|
|
# Append all changes to it
|
|
|
|
for entity_id, group in groupby(states, lambda state: state.entity_id):
|
|
|
|
result[entity_id].extend(group)
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
2015-04-29 02:12:05 +00:00
|
|
|
def get_state(utc_point_in_time, entity_id, run=None):
|
2016-03-07 17:49:31 +00:00
|
|
|
"""Return a state at a specific point in time."""
|
2015-04-29 02:12:05 +00:00
|
|
|
states = get_states(utc_point_in_time, (entity_id,), run)
|
2015-02-07 21:23:01 +00:00
|
|
|
|
|
|
|
return states[0] if states else None
|
2015-02-02 02:00:30 +00:00
|
|
|
|
|
|
|
|
2015-04-07 08:01:23 +00:00
|
|
|
# pylint: disable=unused-argument
|
2015-01-31 18:31:16 +00:00
|
|
|
def setup(hass, config):
|
2016-03-08 16:55:57 +00:00
|
|
|
"""Setup the history hooks."""
|
2015-01-31 18:31:16 +00:00
|
|
|
hass.http.register_path(
|
|
|
|
'GET',
|
2015-02-01 04:06:30 +00:00
|
|
|
re.compile(
|
2015-02-02 02:00:30 +00:00
|
|
|
r'/api/history/entity/(?P<entity_id>[a-zA-Z\._0-9]+)/'
|
|
|
|
r'recent_states'),
|
2015-02-01 04:06:30 +00:00
|
|
|
_api_last_5_states)
|
|
|
|
|
2015-06-16 05:40:57 +00:00
|
|
|
hass.http.register_path('GET', URL_HISTORY_PERIOD, _api_history_period)
|
2015-02-02 02:00:30 +00:00
|
|
|
|
2015-02-01 04:06:30 +00:00
|
|
|
return True
|
2015-01-31 18:31:16 +00:00
|
|
|
|
|
|
|
|
2015-04-07 08:01:23 +00:00
|
|
|
# pylint: disable=unused-argument
|
2015-01-31 18:31:16 +00:00
|
|
|
# pylint: disable=invalid-name
|
|
|
|
def _api_last_5_states(handler, path_match, data):
|
2016-03-07 17:49:31 +00:00
|
|
|
"""Return the last 5 states for an entity id as JSON."""
|
2015-01-31 18:31:16 +00:00
|
|
|
entity_id = path_match.group('entity_id')
|
|
|
|
|
2015-02-07 21:23:01 +00:00
|
|
|
handler.write_json(last_5_states(entity_id))
|
2015-02-02 02:00:30 +00:00
|
|
|
|
|
|
|
|
|
|
|
def _api_history_period(handler, path_match, data):
|
2016-03-07 17:49:31 +00:00
|
|
|
"""Return history over a period of time."""
|
2015-06-16 05:40:57 +00:00
|
|
|
date_str = path_match.group('date')
|
|
|
|
one_day = timedelta(seconds=86400)
|
|
|
|
|
|
|
|
if date_str:
|
|
|
|
start_date = dt_util.date_str_to_date(date_str)
|
|
|
|
|
|
|
|
if start_date is None:
|
|
|
|
handler.write_json_message("Error parsing JSON", HTTP_BAD_REQUEST)
|
|
|
|
return
|
|
|
|
|
|
|
|
start_time = dt_util.as_utc(dt_util.start_of_local_day(start_date))
|
|
|
|
else:
|
|
|
|
start_time = dt_util.utcnow() - one_day
|
|
|
|
|
|
|
|
end_time = start_time + one_day
|
|
|
|
|
2015-02-02 02:00:30 +00:00
|
|
|
entity_id = data.get('filter_entity_id')
|
|
|
|
|
|
|
|
handler.write_json(
|
2016-01-23 20:36:43 +00:00
|
|
|
get_significant_states(start_time, end_time, entity_id).values())
|
2016-03-05 18:28:48 +00:00
|
|
|
|
|
|
|
|
|
|
|
def _is_significant(state):
|
2016-03-08 16:55:57 +00:00
|
|
|
"""Test if state is significant for history charts.
|
2016-03-05 18:28:48 +00:00
|
|
|
|
|
|
|
Will only test for things that are not filtered out in SQL.
|
|
|
|
"""
|
|
|
|
# scripts that are not cancellable will never change state
|
|
|
|
return (state.domain != 'script' or
|
|
|
|
state.attributes.get(script.ATTR_CAN_CANCEL))
|