-
Notifications
You must be signed in to change notification settings - Fork 2
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Showing
11 changed files
with
671 additions
and
45 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -31,3 +31,5 @@ jobs: | |
POETRY_VIRTUALENVS_CREATE: false | ||
- name: Check linting | ||
run: just lint | ||
- name: Run tests | ||
run: just test |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,63 +1,200 @@ | ||
"""Database operations.""" | ||
from datetime import datetime, timedelta | ||
from typing import Optional | ||
from typing import Optional, List | ||
import databases | ||
from dataclasses import dataclass | ||
|
||
|
||
async def get_latest_stats(dbconn: databases.Database): | ||
@dataclass(frozen=True) | ||
class AveragedRead: | ||
"""An averaged read result. | ||
Includes the number of reads that went into the average, as well as the oldest timestamp of those reads. | ||
""" | ||
|
||
avg_pm25: float | ||
avg_pm10: float | ||
count: int | ||
oldest_read_time: datetime | ||
|
||
|
||
@dataclass(frozen=True) | ||
class ReadLogEntry: | ||
"""A read log entry.""" | ||
|
||
event_time: datetime | ||
pm25: float | ||
pm10: float | ||
|
||
|
||
@dataclass(frozen=True) | ||
class EpaAqiLogEntry: | ||
"""An EPA Aqi log entry.""" | ||
|
||
event_time: datetime | ||
epa_aqi: float | ||
pollutant: str | ||
read_count: int | ||
oldest_read_time: datetime | ||
|
||
|
||
async def get_latest_read(dbconn: databases.Database) -> ReadLogEntry: | ||
"""Get the most recent read from the database.""" | ||
result = await dbconn.fetch_one("SELECT * FROM aqi_log ORDER BY event_time DESC LIMIT 1") | ||
result = await dbconn.fetch_one("SELECT event_time, pm25, pm10 FROM read_log ORDER BY event_time DESC LIMIT 1") | ||
if result: | ||
return result[0], result[1], result[2], result[3] | ||
return ReadLogEntry(datetime.fromtimestamp(result[0]), result[1], result[2]) | ||
else: | ||
return ReadLogEntry(datetime.now(), 0, 0) | ||
|
||
|
||
async def get_latest_epa_aqi(dbconn: databases.Database) -> EpaAqiLogEntry: | ||
"""Get the most recent EPA AQI from the database.""" | ||
result = await dbconn.fetch_one( | ||
"SELECT event_time, epa_aqi, pollutant, read_count, oldest_read_time " | ||
"FROM epa_aqi_log ORDER BY event_time DESC LIMIT 1" | ||
) | ||
if result: | ||
return EpaAqiLogEntry( | ||
event_time=datetime.fromtimestamp(result[0]), | ||
epa_aqi=result[1], | ||
pollutant=result[2], | ||
read_count=result[3], | ||
oldest_read_time=datetime.fromtimestamp(result[4]), | ||
) | ||
else: | ||
return EpaAqiLogEntry( | ||
event_time=datetime.now(), epa_aqi=0, pollutant="NA", read_count=0, oldest_read_time=datetime.now() | ||
) | ||
|
||
|
||
async def get_all_reads(dbconn: databases.Database, lookback: Optional[datetime]) -> List[ReadLogEntry]: | ||
"""Retrieve all read stats, for a given time window. | ||
If no window is specified, all results are returned. | ||
""" | ||
if lookback: | ||
data = await dbconn.fetch_all( | ||
"SELECT event_time, pm10, pm25 FROM read_log WHERE event_time >= :lookback ORDER BY event_time ASC", | ||
values={"lookback": int(lookback.timestamp())}, | ||
) | ||
else: | ||
return 0, 0, 0, 0 | ||
data = await dbconn.fetch_all("SELECT event_time, pm10, pm25 FROM read_log ORDER BY event_time ASC") | ||
|
||
return [ReadLogEntry(event_time=datetime.fromtimestamp(x[0]), pm10=x[1], pm25=x[2]) for x in data] | ||
|
||
async def get_all_stats(dbconn: databases.Database, window_delta: Optional[timedelta]): | ||
|
||
async def get_all_epa_aqis(dbconn: databases.Database, lookback: Optional[datetime]) -> List[EpaAqiLogEntry]: | ||
"""Retrieve all read stats, for a given time window. | ||
If no window is specified, all results are returned. | ||
""" | ||
if window_delta: | ||
lookback = int((datetime.now() - window_delta).timestamp()) | ||
return await dbconn.fetch_all( | ||
"SELECT * FROM aqi_log WHERE event_time >= :lookback ORDER BY event_time ASC", | ||
values={"lookback": lookback}, | ||
if lookback: | ||
data = await dbconn.fetch_all( | ||
"SELECT event_time, epa_aqi, pollutant, read_count, oldest_read_time " | ||
"FROM epa_aqi_log " | ||
"WHERE event_time >= :lookback ORDER BY event_time ASC", | ||
values={"lookback": int(lookback.timestamp())}, | ||
) | ||
else: | ||
return await dbconn.fetch_all("SELECT * FROM aqi_log ORDER BY event_time ASC") | ||
data = await dbconn.fetch_all( | ||
"SELECT event_time, epa_aqi, pollutant, read_count, oldest_read_time " | ||
"FROM epa_aqi_log ORDER BY event_time ASC" | ||
) | ||
|
||
return [ | ||
EpaAqiLogEntry( | ||
event_time=datetime.fromtimestamp(x[0]), | ||
epa_aqi=x[1], | ||
pollutant=x[2], | ||
read_count=x[3], | ||
oldest_read_time=datetime.fromtimestamp(x[4]), | ||
) | ||
for x in data | ||
] | ||
|
||
|
||
async def get_averaged_reads(dbconn: databases.Database, lookback_to: datetime) -> Optional[AveragedRead]: | ||
"""Get the average read values, looking back to a certain time. | ||
Note that the lookback will include one additional value outside of the window if it exists. This allows for us to | ||
ensure full coverage of the lookback window. | ||
""" | ||
lookback = int(lookback_to.timestamp()) | ||
result = await dbconn.fetch_one( | ||
"SELECT " | ||
"AVG(pm25) as avg_pm25, AVG(pm10) as avg_pm10, COUNT(*) as count, MIN(event_time) as oldest_time " | ||
"FROM read_log " | ||
"WHERE (event_time >= :lookback) OR " | ||
"(event_time = (SELECT MAX(event_time) FROM read_log WHERE event_time <= :lookback)) ORDER BY event_time ASC", | ||
values={"lookback": lookback}, | ||
) | ||
|
||
async def clean_old(dbconn: databases.Database, retention_minutes: int): | ||
if result is None: | ||
return None | ||
else: | ||
return AveragedRead( | ||
avg_pm25=result[0], | ||
avg_pm10=result[1], | ||
count=result[2], | ||
oldest_read_time=datetime.fromtimestamp(result[3]), | ||
) | ||
|
||
|
||
async def clean_old(dbconn: databases.Database, retention_minutes: int) -> None: | ||
"""Remove expired database entries. | ||
This is used to keep the database from going infinitely, and allows us to define a retention period. | ||
""" | ||
last_week = datetime.now() - timedelta(minutes=retention_minutes) | ||
last_week_timestamp = int(last_week.timestamp()) | ||
await dbconn.execute( | ||
"DELETE FROM aqi_log WHERE event_time < :last_week_timestamp", | ||
"DELETE FROM read_log WHERE event_time < :last_week_timestamp", | ||
values={"last_week_timestamp": last_week_timestamp}, | ||
) | ||
|
||
|
||
async def add_entry(dbconn: databases.Database, event_time, epa_aqi_pm25, raw_pm25, raw_pm10): | ||
"""Add a read entry to the database.""" | ||
async def add_epa_read( | ||
dbconn: databases.Database, | ||
event_time: datetime, | ||
epa_aqi: float, | ||
pollutant: str, | ||
read_count: int, | ||
oldest_read_time: datetime, | ||
): | ||
"""Add an EPA read entry to the database.""" | ||
formatted_time = int(event_time.timestamp()) | ||
formatted_oldest_read_time = int(oldest_read_time.timestamp()) | ||
await dbconn.execute( | ||
query="INSERT INTO epa_aqi_log VALUES (:formatted_time, :epa_aqi, :pollutant, :read_count, :oldest_read_time)", | ||
values={ | ||
"formatted_time": formatted_time, | ||
"epa_aqi": epa_aqi, | ||
"pollutant": pollutant, | ||
"read_count": read_count, | ||
"oldest_read_time": formatted_oldest_read_time, | ||
}, | ||
) | ||
|
||
|
||
async def add_read(dbconn: databases.Database, event_time: datetime, pm25: float, pm10: float): | ||
"""Add a raw read entry to the database.""" | ||
formatted_time = int(event_time.timestamp()) | ||
await dbconn.execute( | ||
query="INSERT INTO aqi_log VALUES (:formatted_time, :epa_aqi_pm25, :raw_pm25, :raw_pm10)", | ||
query="INSERT INTO read_log VALUES (:formatted_time, :pm25, :pm10)", | ||
values={ | ||
"formatted_time": formatted_time, | ||
"epa_aqi_pm25": epa_aqi_pm25, | ||
"raw_pm25": raw_pm25, | ||
"raw_pm10": raw_pm10, | ||
"pm25": pm25, | ||
"pm10": pm10, | ||
}, | ||
) | ||
|
||
|
||
async def create_tables(dbconn: databases.Database): | ||
"""Create database tables, if they don't already exist.""" | ||
await dbconn.execute("""CREATE TABLE IF NOT EXISTS read_log (event_time integer, pm25 real, pm10 real)""") | ||
await dbconn.execute("""CREATE INDEX IF NOT EXISTS read_eventtime ON read_log (event_time)""") | ||
await dbconn.execute( | ||
"""CREATE TABLE IF NOT EXISTS aqi_log (event_time integer, epa_aqi_pm25 real, raw_pm25 real, raw_pm10 real)""" | ||
"""CREATE TABLE IF NOT EXISTS epa_aqi_log | ||
(event_time integer, epa_aqi real, pollutant text, read_count integer, oldest_read_time integer)""" | ||
) | ||
await dbconn.execute("""CREATE INDEX IF NOT EXISTS aqi_eventtime ON aqi_log (event_time)""") | ||
await dbconn.execute("""CREATE INDEX IF NOT EXISTS eqpaqi_eventtime ON epa_aqi_log (event_time)""") |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,282 @@ | ||
import pytest | ||
import pytest_asyncio | ||
import databases | ||
|
||
from aqimon import database | ||
from datetime import timedelta, datetime | ||
|
||
|
||
@pytest_asyncio.fixture | ||
async def database_conn(): | ||
"""Fixture to set up the in-memory database with test data.""" | ||
dbconn = databases.Database("sqlite+aiosqlite://:memory:", force_rollback=True) | ||
await dbconn.connect() | ||
await database.create_tables(dbconn) | ||
yield dbconn | ||
await dbconn.disconnect() | ||
|
||
|
||
@pytest.mark.asyncio | ||
async def test_get_latest_read(database_conn): | ||
current_time = datetime(2020, 1, 1, 12, 0, 0) | ||
await database.add_read(database_conn, current_time - timedelta(hours=2), pm10=1, pm25=2) | ||
await database.add_read(database_conn, current_time - timedelta(hours=4), pm10=2, pm25=3) | ||
|
||
result = await database.get_latest_read(database_conn) | ||
assert result.pm10 == 1.0 | ||
assert result.pm25 == 2.0 | ||
assert result.event_time == current_time - timedelta(hours=2) | ||
|
||
|
||
@pytest.mark.asyncio | ||
async def test_get_latest_read_no_data(database_conn): | ||
result = await database.get_latest_read(database_conn) | ||
assert result.pm10 == 0.0 | ||
assert result.pm25 == 0.0 | ||
assert result.event_time is not None | ||
|
||
|
||
@pytest.mark.asyncio | ||
async def test_get_latest_epa_aqi(database_conn): | ||
current_time = datetime(2020, 1, 1, 12, 0, 0) | ||
await database.add_epa_read( | ||
database_conn, | ||
current_time - timedelta(hours=2), | ||
epa_aqi=3.5, | ||
read_count=3, | ||
pollutant="PM25", | ||
oldest_read_time=current_time - timedelta(days=3), | ||
) | ||
await database.add_epa_read( | ||
database_conn, | ||
current_time - timedelta(hours=4), | ||
epa_aqi=3.7, | ||
read_count=5, | ||
pollutant="PM25", | ||
oldest_read_time=current_time - timedelta(days=3), | ||
) | ||
|
||
result = await database.get_latest_epa_aqi(database_conn) | ||
assert result.epa_aqi == 3.5 | ||
assert result.event_time == current_time - timedelta(hours=2) | ||
|
||
|
||
@pytest.mark.asyncio | ||
async def test_get_latest_epq_aqi_no_data(database_conn): | ||
result = await database.get_latest_epa_aqi(database_conn) | ||
assert result.epa_aqi == 0 | ||
assert result.event_time is not None | ||
|
||
|
||
@pytest.mark.asyncio | ||
async def test_get_all_reads(database_conn): | ||
current_time = datetime(2020, 1, 1, 12, 0, 0) | ||
await database.add_read(database_conn, current_time - timedelta(hours=1), pm10=1, pm25=2) | ||
await database.add_read(database_conn, current_time - timedelta(hours=2), pm10=2, pm25=3) | ||
await database.add_read(database_conn, current_time - timedelta(hours=3), pm10=3, pm25=4) | ||
|
||
result = await database.get_all_reads(database_conn, lookback=None) | ||
assert len(result) == 3 | ||
assert result[2].pm10 == 1.0 | ||
assert result[2].pm25 == 2.0 | ||
assert result[2].event_time == current_time - timedelta(hours=1) | ||
|
||
assert result[0].pm10 == 3.0 | ||
assert result[0].pm25 == 4.0 | ||
assert result[0].event_time == current_time - timedelta(hours=3) | ||
|
||
|
||
@pytest.mark.asyncio | ||
async def test_get_all_reads_with_window(database_conn): | ||
current_time = datetime(2020, 1, 1, 12, 0, 0) | ||
await database.add_read(database_conn, current_time - timedelta(hours=1), pm10=1, pm25=2) | ||
await database.add_read(database_conn, current_time - timedelta(hours=2), pm10=2, pm25=3) | ||
await database.add_read(database_conn, current_time - timedelta(hours=3), pm10=3, pm25=4) | ||
|
||
result = await database.get_all_reads(database_conn, current_time - timedelta(hours=2, minutes=30)) | ||
assert len(result) == 2 | ||
assert result[1].pm10 == 1.0 | ||
assert result[1].pm25 == 2.0 | ||
assert result[1].event_time == current_time - timedelta(hours=1) | ||
|
||
assert result[0].pm10 == 2.0 | ||
assert result[0].pm25 == 3.0 | ||
assert result[0].event_time == current_time - timedelta(hours=2) | ||
|
||
|
||
@pytest.mark.asyncio | ||
async def test_get_all_epa_aqi(database_conn): | ||
current_time = datetime(2020, 1, 1, 12, 0, 0) | ||
await database.add_epa_read( | ||
database_conn, | ||
current_time - timedelta(hours=1), | ||
epa_aqi=2, | ||
pollutant="PM25", | ||
read_count=5, | ||
oldest_read_time=current_time - timedelta(days=3), | ||
) | ||
await database.add_epa_read( | ||
database_conn, | ||
current_time - timedelta(hours=2), | ||
epa_aqi=3, | ||
pollutant="PM10", | ||
read_count=20, | ||
oldest_read_time=current_time - timedelta(days=60), | ||
) | ||
await database.add_epa_read( | ||
database_conn, | ||
current_time - timedelta(hours=3), | ||
epa_aqi=4, | ||
pollutant="PM25", | ||
read_count=10, | ||
oldest_read_time=current_time - timedelta(days=30), | ||
) | ||
|
||
result = await database.get_all_epa_aqis(database_conn, lookback=None) | ||
assert len(result) == 3 | ||
assert result[2].epa_aqi == 2.0 | ||
assert result[2].read_count == 5 | ||
assert result[2].pollutant == "PM25" | ||
assert result[2].oldest_read_time == current_time - timedelta(days=3) | ||
assert result[2].event_time == current_time - timedelta(hours=1) | ||
|
||
assert result[0].epa_aqi == 4.0 | ||
assert result[0].read_count == 10 | ||
assert result[0].pollutant == "PM25" | ||
assert result[0].oldest_read_time == current_time - timedelta(days=30) | ||
assert result[0].event_time == current_time - timedelta(hours=3) | ||
|
||
|
||
@pytest.mark.asyncio | ||
async def test_get_all_epa_aqi_with_window(database_conn): | ||
current_time = datetime(2020, 1, 1, 12, 0, 0) | ||
await database.add_epa_read( | ||
database_conn, | ||
current_time - timedelta(hours=1), | ||
epa_aqi=2, | ||
pollutant="PM25", | ||
read_count=5, | ||
oldest_read_time=current_time - timedelta(days=3), | ||
) | ||
await database.add_epa_read( | ||
database_conn, | ||
current_time - timedelta(hours=2), | ||
epa_aqi=3, | ||
pollutant="PM10", | ||
read_count=20, | ||
oldest_read_time=current_time - timedelta(days=60), | ||
) | ||
await database.add_epa_read( | ||
database_conn, | ||
current_time - timedelta(hours=3), | ||
epa_aqi=4, | ||
pollutant="PM25", | ||
read_count=10, | ||
oldest_read_time=current_time - timedelta(days=30), | ||
) | ||
|
||
result = await database.get_all_epa_aqis(database_conn, current_time - timedelta(hours=2, minutes=30)) | ||
assert len(result) == 2 | ||
assert result[1].epa_aqi == 2.0 | ||
assert result[1].read_count == 5 | ||
assert result[1].pollutant == "PM25" | ||
assert result[1].oldest_read_time == current_time - timedelta(days=3) | ||
assert result[1].event_time == current_time - timedelta(hours=1) | ||
|
||
assert result[0].epa_aqi == 3.0 | ||
assert result[0].read_count == 20 | ||
assert result[0].pollutant == "PM10" | ||
assert result[0].oldest_read_time == current_time - timedelta(days=60) | ||
assert result[0].event_time == current_time - timedelta(hours=2) | ||
|
||
|
||
@pytest.mark.asyncio | ||
async def test_get_averaged_reads(database_conn): | ||
# Add reads every two hours | ||
current_time = datetime(2020, 1, 1, 12, 0, 0) | ||
lookback_to = current_time - timedelta(hours=8) | ||
await database.add_read(database_conn, current_time - timedelta(hours=2), pm10=1, pm25=2) | ||
await database.add_read(database_conn, current_time - timedelta(hours=4), pm10=2, pm25=3) | ||
await database.add_read(database_conn, current_time - timedelta(hours=6), pm10=3, pm25=4) | ||
await database.add_read(database_conn, current_time - timedelta(hours=8), pm10=4, pm25=5) | ||
|
||
result = await database.get_averaged_reads(database_conn, lookback_to) | ||
assert result.count == 4 | ||
assert result.avg_pm10 == 2.5 | ||
assert result.avg_pm25 == 3.5 | ||
assert result.oldest_read_time == current_time - timedelta(hours=8) | ||
|
||
|
||
@pytest.mark.asyncio | ||
async def test_get_averaged_reads_looks_past(database_conn): | ||
current_time = datetime(2020, 1, 1, 12, 0, 0) | ||
lookback_to = current_time - timedelta(hours=8) | ||
await database.add_read(database_conn, current_time - timedelta(hours=6), pm10=1, pm25=2) | ||
await database.add_read(database_conn, current_time - timedelta(hours=7), pm10=2, pm25=3) | ||
# Should be included since its the read just after the lookback | ||
await database.add_read(database_conn, current_time - timedelta(hours=8, minutes=5), pm10=3, pm25=4) | ||
# Should be excluded | ||
await database.add_read(database_conn, current_time - timedelta(hours=9), pm10=4, pm25=5) | ||
|
||
result = await database.get_averaged_reads(database_conn, lookback_to) | ||
assert result.count == 3 | ||
assert result.avg_pm10 == 2.0 | ||
assert result.avg_pm25 == 3.0 | ||
assert result.oldest_read_time == current_time - timedelta(hours=8, minutes=5) | ||
|
||
|
||
@pytest.mark.asyncio | ||
async def test_clean_old(database_conn): | ||
current_time = datetime.now() | ||
await database.add_read(database_conn, current_time - timedelta(hours=2), pm10=1, pm25=2) | ||
await database.add_read(database_conn, current_time - timedelta(hours=4), pm10=2, pm25=3) | ||
# These should be deleted | ||
await database.add_read(database_conn, current_time - timedelta(hours=6), pm10=3, pm25=4) | ||
await database.add_read(database_conn, current_time - timedelta(hours=8), pm10=4, pm25=5) | ||
|
||
await database.clean_old(database_conn, retention_minutes=(60 * 4) + 30) | ||
|
||
result = await database.get_all_reads(database_conn, lookback=None) | ||
assert len(result) == 2 | ||
assert result[0].event_time == (current_time - timedelta(hours=4)).replace(microsecond=0) | ||
assert result[1].event_time == (current_time - timedelta(hours=2)).replace(microsecond=0) | ||
|
||
|
||
@pytest.mark.asyncio | ||
async def test_add_read(database_conn): | ||
current_time = datetime.now() | ||
|
||
await database.add_read(database_conn, current_time - timedelta(hours=2), pm10=1, pm25=2) | ||
await database.add_read(database_conn, current_time - timedelta(hours=4), pm10=2, pm25=3) | ||
|
||
result = await database.get_all_reads(database_conn, lookback=None) | ||
assert len(result) == 2 | ||
assert result[0].event_time == (current_time - timedelta(hours=4)).replace(microsecond=0) | ||
assert result[1].event_time == (current_time - timedelta(hours=2)).replace(microsecond=0) | ||
|
||
|
||
@pytest.mark.asyncio | ||
async def test_add_epa_read(database_conn): | ||
current_time = datetime.now() | ||
|
||
await database.add_epa_read( | ||
database_conn, | ||
current_time - timedelta(hours=2), | ||
epa_aqi=2, | ||
pollutant="PM25", | ||
read_count=5, | ||
oldest_read_time=current_time - timedelta(days=3), | ||
) | ||
await database.add_epa_read( | ||
database_conn, | ||
current_time - timedelta(hours=4), | ||
epa_aqi=3, | ||
pollutant="PM10", | ||
read_count=20, | ||
oldest_read_time=current_time - timedelta(days=60), | ||
) | ||
|
||
result = await database.get_all_epa_aqis(database_conn, lookback=None) | ||
assert len(result) == 2 | ||
assert result[0].event_time == (current_time - timedelta(hours=4)).replace(microsecond=0) | ||
assert result[1].event_time == (current_time - timedelta(hours=2)).replace(microsecond=0) |