2017-05-31 04:34:40 +00:00
|
|
|
"""Script to run benchmarks."""
|
|
|
|
import argparse
|
2018-01-21 06:35:38 +00:00
|
|
|
import asyncio
|
2017-05-31 04:34:40 +00:00
|
|
|
from contextlib import suppress
|
2017-10-10 20:26:03 +00:00
|
|
|
from datetime import datetime
|
2017-05-31 04:34:40 +00:00
|
|
|
import logging
|
|
|
|
from timeit import default_timer as timer
|
|
|
|
|
|
|
|
from homeassistant import core
|
2018-01-21 06:35:38 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
ATTR_NOW, EVENT_STATE_CHANGED, EVENT_TIME_CHANGED)
|
2017-10-10 20:26:03 +00:00
|
|
|
from homeassistant.util import dt as dt_util
|
2017-05-31 04:34:40 +00:00
|
|
|
|
|
|
|
BENCHMARKS = {}
|
|
|
|
|
|
|
|
|
|
|
|
def run(args):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Handle ensure configuration commandline script."""
|
2017-05-31 04:34:40 +00:00
|
|
|
# Disable logging
|
|
|
|
logging.getLogger('homeassistant.core').setLevel(logging.CRITICAL)
|
|
|
|
|
|
|
|
parser = argparse.ArgumentParser(
|
|
|
|
description=("Run a Home Assistant benchmark."))
|
|
|
|
parser.add_argument('name', choices=BENCHMARKS)
|
|
|
|
parser.add_argument('--script', choices=['benchmark'])
|
|
|
|
|
|
|
|
args = parser.parse_args()
|
|
|
|
|
|
|
|
bench = BENCHMARKS[args.name]
|
|
|
|
|
|
|
|
print('Using event loop:', asyncio.get_event_loop_policy().__module__)
|
|
|
|
|
|
|
|
with suppress(KeyboardInterrupt):
|
|
|
|
while True:
|
|
|
|
loop = asyncio.new_event_loop()
|
|
|
|
hass = core.HomeAssistant(loop)
|
|
|
|
hass.async_stop_track_tasks()
|
|
|
|
runtime = loop.run_until_complete(bench(hass))
|
|
|
|
print('Benchmark {} done in {}s'.format(bench.__name__, runtime))
|
|
|
|
loop.run_until_complete(hass.async_stop())
|
|
|
|
loop.close()
|
|
|
|
|
|
|
|
return 0
|
|
|
|
|
|
|
|
|
|
|
|
def benchmark(func):
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Decorate to mark a benchmark."""
|
2017-05-31 04:34:40 +00:00
|
|
|
BENCHMARKS[func.__name__] = func
|
|
|
|
return func
|
|
|
|
|
|
|
|
|
|
|
|
@benchmark
|
2018-02-25 11:38:46 +00:00
|
|
|
async def async_million_events(hass):
|
2017-05-31 04:34:40 +00:00
|
|
|
"""Run a million events."""
|
|
|
|
count = 0
|
|
|
|
event_name = 'benchmark_event'
|
|
|
|
event = asyncio.Event(loop=hass.loop)
|
|
|
|
|
|
|
|
@core.callback
|
|
|
|
def listener(_):
|
|
|
|
"""Handle event."""
|
|
|
|
nonlocal count
|
|
|
|
count += 1
|
|
|
|
|
|
|
|
if count == 10**6:
|
|
|
|
event.set()
|
|
|
|
|
|
|
|
hass.bus.async_listen(event_name, listener)
|
|
|
|
|
2017-10-10 20:26:03 +00:00
|
|
|
for _ in range(10**6):
|
|
|
|
hass.bus.async_fire(event_name)
|
|
|
|
|
2017-05-31 04:34:40 +00:00
|
|
|
start = timer()
|
|
|
|
|
2018-02-25 11:38:46 +00:00
|
|
|
await event.wait()
|
2017-10-10 20:26:03 +00:00
|
|
|
|
|
|
|
return timer() - start
|
|
|
|
|
|
|
|
|
|
|
|
@benchmark
|
2018-02-25 11:38:46 +00:00
|
|
|
async def async_million_time_changed_helper(hass):
|
2017-10-10 20:26:03 +00:00
|
|
|
"""Run a million events through time changed helper."""
|
|
|
|
count = 0
|
|
|
|
event = asyncio.Event(loop=hass.loop)
|
|
|
|
|
|
|
|
@core.callback
|
|
|
|
def listener(_):
|
|
|
|
"""Handle event."""
|
|
|
|
nonlocal count
|
|
|
|
count += 1
|
|
|
|
|
|
|
|
if count == 10**6:
|
|
|
|
event.set()
|
|
|
|
|
|
|
|
hass.helpers.event.async_track_time_change(listener, minute=0, second=0)
|
|
|
|
event_data = {
|
|
|
|
ATTR_NOW: datetime(2017, 10, 10, 15, 0, 0, tzinfo=dt_util.UTC)
|
|
|
|
}
|
|
|
|
|
2017-05-31 04:34:40 +00:00
|
|
|
for _ in range(10**6):
|
2017-10-10 20:26:03 +00:00
|
|
|
hass.bus.async_fire(EVENT_TIME_CHANGED, event_data)
|
|
|
|
|
|
|
|
start = timer()
|
|
|
|
|
2018-02-25 11:38:46 +00:00
|
|
|
await event.wait()
|
2017-10-10 20:26:03 +00:00
|
|
|
|
|
|
|
return timer() - start
|
|
|
|
|
|
|
|
|
|
|
|
@benchmark
|
2018-02-25 11:38:46 +00:00
|
|
|
async def async_million_state_changed_helper(hass):
|
2017-10-10 20:26:03 +00:00
|
|
|
"""Run a million events through state changed helper."""
|
|
|
|
count = 0
|
|
|
|
entity_id = 'light.kitchen'
|
|
|
|
event = asyncio.Event(loop=hass.loop)
|
|
|
|
|
|
|
|
@core.callback
|
|
|
|
def listener(*args):
|
|
|
|
"""Handle event."""
|
|
|
|
nonlocal count
|
|
|
|
count += 1
|
|
|
|
|
|
|
|
if count == 10**6:
|
|
|
|
event.set()
|
|
|
|
|
|
|
|
hass.helpers.event.async_track_state_change(
|
|
|
|
entity_id, listener, 'off', 'on')
|
|
|
|
event_data = {
|
|
|
|
'entity_id': entity_id,
|
|
|
|
'old_state': core.State(entity_id, 'off'),
|
|
|
|
'new_state': core.State(entity_id, 'on'),
|
|
|
|
}
|
|
|
|
|
|
|
|
for _ in range(10**6):
|
|
|
|
hass.bus.async_fire(EVENT_STATE_CHANGED, event_data)
|
|
|
|
|
|
|
|
start = timer()
|
2017-05-31 04:34:40 +00:00
|
|
|
|
2018-02-25 11:38:46 +00:00
|
|
|
await event.wait()
|
2017-05-31 04:34:40 +00:00
|
|
|
|
|
|
|
return timer() - start
|
2018-02-21 18:35:55 +00:00
|
|
|
|
|
|
|
|
|
|
|
@benchmark
|
|
|
|
@asyncio.coroutine
|
|
|
|
def logbook_filtering_state(hass):
|
|
|
|
"""Filter state changes."""
|
|
|
|
return _logbook_filtering(hass, 1, 1)
|
|
|
|
|
|
|
|
|
|
|
|
@benchmark
|
|
|
|
@asyncio.coroutine
|
|
|
|
def logbook_filtering_attributes(hass):
|
|
|
|
"""Filter attribute changes."""
|
|
|
|
return _logbook_filtering(hass, 1, 2)
|
|
|
|
|
|
|
|
|
|
|
|
@benchmark
|
|
|
|
@asyncio.coroutine
|
|
|
|
def _logbook_filtering(hass, last_changed, last_updated):
|
|
|
|
from homeassistant.components import logbook
|
|
|
|
|
|
|
|
entity_id = 'test.entity'
|
|
|
|
|
|
|
|
old_state = {
|
|
|
|
'entity_id': entity_id,
|
|
|
|
'state': 'off'
|
|
|
|
}
|
|
|
|
|
|
|
|
new_state = {
|
|
|
|
'entity_id': entity_id,
|
|
|
|
'state': 'on',
|
|
|
|
'last_updated': last_updated,
|
|
|
|
'last_changed': last_changed
|
|
|
|
}
|
|
|
|
|
|
|
|
event = core.Event(EVENT_STATE_CHANGED, {
|
|
|
|
'entity_id': entity_id,
|
|
|
|
'old_state': old_state,
|
|
|
|
'new_state': new_state
|
|
|
|
})
|
|
|
|
|
|
|
|
events = [event] * 10**5
|
|
|
|
|
|
|
|
start = timer()
|
|
|
|
|
|
|
|
# pylint: disable=protected-access
|
|
|
|
events = logbook._exclude_events(events, {})
|
2018-10-01 14:12:25 +00:00
|
|
|
list(logbook.humanify(None, events))
|
2018-02-21 18:35:55 +00:00
|
|
|
|
|
|
|
return timer() - start
|