"""Test the ORM's `Forecast` model.""" 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()