Alexander Hess
e8c97dd7da
- the model handles the caching of demand forecasting results - include the database migration script
147 lines
4.6 KiB
Python
147 lines
4.6 KiB
Python
"""Test the ORM's `Forecast` model."""
|
|
# pylint:disable=no-self-use
|
|
|
|
import datetime
|
|
|
|
import pytest
|
|
import sqlalchemy as sqla
|
|
from sqlalchemy import exc as sa_exc
|
|
|
|
from urban_meal_delivery import db
|
|
|
|
|
|
@pytest.fixture
|
|
def forecast(pixel):
|
|
"""A `forecast` made in the `pixel`."""
|
|
return db.Forecast(
|
|
pixel=pixel,
|
|
start_at=datetime.datetime(2020, 1, 1, 12, 0),
|
|
time_step=60,
|
|
training_horizon=8,
|
|
method='hets',
|
|
prediction=12.3,
|
|
)
|
|
|
|
|
|
class TestSpecialMethods:
|
|
"""Test special methods in `Forecast`."""
|
|
|
|
def test_create_forecast(self, forecast):
|
|
"""Test instantiation of a new `Forecast` object."""
|
|
assert forecast is not None
|
|
|
|
|
|
@pytest.mark.db
|
|
@pytest.mark.no_cover
|
|
class TestConstraints:
|
|
"""Test the database constraints defined in `Forecast`."""
|
|
|
|
def test_insert_into_database(self, db_session, forecast):
|
|
"""Insert an instance into the (empty) database."""
|
|
assert db_session.query(db.Forecast).count() == 0
|
|
|
|
db_session.add(forecast)
|
|
db_session.commit()
|
|
|
|
assert db_session.query(db.Forecast).count() == 1
|
|
|
|
def test_delete_a_referenced_pixel(self, db_session, forecast):
|
|
"""Remove a record that is referenced with a FK."""
|
|
db_session.add(forecast)
|
|
db_session.commit()
|
|
|
|
# Must delete without ORM as otherwise an UPDATE statement is emitted.
|
|
stmt = sqla.delete(db.Pixel).where(db.Pixel.id == forecast.pixel.id)
|
|
|
|
with pytest.raises(
|
|
sa_exc.IntegrityError, match='fk_forecasts_to_pixels_via_pixel_id',
|
|
):
|
|
db_session.execute(stmt)
|
|
|
|
@pytest.mark.parametrize('hour', [10, 23])
|
|
def test_invalid_start_at_outside_operating_hours(
|
|
self, db_session, forecast, hour,
|
|
):
|
|
"""Insert an instance with invalid data."""
|
|
forecast.start_at = datetime.datetime(
|
|
forecast.start_at.year,
|
|
forecast.start_at.month,
|
|
forecast.start_at.day,
|
|
hour,
|
|
)
|
|
db_session.add(forecast)
|
|
|
|
with pytest.raises(
|
|
sa_exc.IntegrityError, match='within_operating_hours',
|
|
):
|
|
db_session.commit()
|
|
|
|
def test_invalid_start_at_not_quarter_of_hour(self, db_session, forecast):
|
|
"""Insert an instance with invalid data."""
|
|
forecast.start_at += datetime.timedelta(minutes=1)
|
|
db_session.add(forecast)
|
|
|
|
with pytest.raises(
|
|
sa_exc.IntegrityError, match='must_be_quarters_of_the_hour',
|
|
):
|
|
db_session.commit()
|
|
|
|
def test_invalid_start_at_seconds_set(self, db_session, forecast):
|
|
"""Insert an instance with invalid data."""
|
|
forecast.start_at += datetime.timedelta(seconds=1)
|
|
db_session.add(forecast)
|
|
|
|
with pytest.raises(
|
|
sa_exc.IntegrityError, match='no_seconds',
|
|
):
|
|
db_session.commit()
|
|
|
|
def test_invalid_start_at_microseconds_set(self, db_session, forecast):
|
|
"""Insert an instance with invalid data."""
|
|
forecast.start_at += datetime.timedelta(microseconds=1)
|
|
db_session.add(forecast)
|
|
|
|
with pytest.raises(
|
|
sa_exc.IntegrityError, match='no_microseconds',
|
|
):
|
|
db_session.commit()
|
|
|
|
@pytest.mark.parametrize('value', [-1, 0])
|
|
def test_positive_time_step(self, db_session, forecast, value):
|
|
"""Insert an instance with invalid data."""
|
|
forecast.time_step = value
|
|
db_session.add(forecast)
|
|
|
|
with pytest.raises(
|
|
sa_exc.IntegrityError, match='time_step_must_be_positive',
|
|
):
|
|
db_session.commit()
|
|
|
|
@pytest.mark.parametrize('value', [-1, 0])
|
|
def test_positive_training_horizon(self, db_session, forecast, value):
|
|
"""Insert an instance with invalid data."""
|
|
forecast.training_horizon = value
|
|
db_session.add(forecast)
|
|
|
|
with pytest.raises(
|
|
sa_exc.IntegrityError, match='training_horizon_must_be_positive',
|
|
):
|
|
db_session.commit()
|
|
|
|
def test_two_predictions_for_same_forecasting_setting(self, db_session, forecast):
|
|
"""Insert a record that violates a unique constraint."""
|
|
db_session.add(forecast)
|
|
db_session.commit()
|
|
|
|
another_forecast = db.Forecast(
|
|
pixel=forecast.pixel,
|
|
start_at=forecast.start_at,
|
|
time_step=forecast.time_step,
|
|
training_horizon=forecast.training_horizon,
|
|
method=forecast.method,
|
|
prediction=99.9,
|
|
)
|
|
db_session.add(another_forecast)
|
|
|
|
with pytest.raises(sa_exc.IntegrityError, match='duplicate key value'):
|
|
db_session.commit()
|