import warnings
from typing import Dict
import numpy as np
import pandas as pd
from etna.datasets import TSDataset
from etna.distributions import BaseDistribution
from etna.distributions import IntDistribution
from etna.models.base import NonPredictionIntervalContextRequiredAbstractModel
[docs]class SeasonalMovingAverageModel(
NonPredictionIntervalContextRequiredAbstractModel,
):
"""Seasonal moving average.
.. math::
y_{t} = \\frac{\\sum_{i=1}^{n} y_{t-is} }{n},
where :math:`s` is seasonality, :math:`n` is window size (how many history values are taken for forecast).
Notes
-----
This model supports in-sample and out-of-sample prediction decomposition.
Prediction components are corresponding target lags with weights of :math:`1/window`.
"""
def __init__(self, window: int = 5, seasonality: int = 7):
"""
Initialize seasonal moving average model.
Length of the context is ``window * seasonality``.
Parameters
----------
window:
Number of values taken for forecast for each point.
seasonality:
Lag between values taken for forecast.
"""
self.window = window
self.seasonality = seasonality
@property
def context_size(self) -> int:
"""Context size of the model."""
return self.window * self.seasonality
[docs] def get_model(self) -> "SeasonalMovingAverageModel":
"""Get internal model.
Returns
-------
:
Itself
"""
return self
[docs] def fit(self, ts: TSDataset) -> "SeasonalMovingAverageModel":
"""Fit model.
For this model, fit does nothing.
Parameters
----------
ts:
Dataset with features
Returns
-------
:
Model after fit
"""
columns = set(ts.columns.get_level_values("feature"))
if columns != {"target"}:
warnings.warn(
message=f"{type(self).__name__} does not work with any exogenous series or features. "
f"It uses only target series for predict/\n "
)
return self
def _validate_context(self, df: pd.DataFrame, prediction_size: int):
"""Validate that we have enough context to make prediction with given parameters."""
expected_length = prediction_size + self.context_size
if len(df) < expected_length:
raise ValueError(
"Given context isn't big enough, try to decrease context_size, prediction_size or increase length of given dataframe!"
)
def _predict_components(self, df: pd.DataFrame, prediction_size: int) -> pd.DataFrame:
"""Estimate forecast components.
Parameters
----------
df:
DatŠ°Frame with target, containing lags that was used to make a prediction
prediction_size:
Number of last timestamps to leave after making prediction.
Previous timestamps will be used as a context.
Returns
-------
:
DataFrame with target components
"""
self._validate_context(df=df, prediction_size=prediction_size)
all_transformed_features = []
segments = sorted(set(df.columns.get_level_values("segment")))
lags = list(range(self.seasonality, self.context_size + 1, self.seasonality))
target = df.loc[:, pd.IndexSlice[:, "target"]]
for lag in lags:
transformed_features = target.shift(lag)
transformed_features.columns = pd.MultiIndex.from_product(
[segments, [f"target_component_lag_{lag}"]], names=("segment", "feature")
)
all_transformed_features.append(transformed_features)
target_components_df = pd.concat(all_transformed_features, axis=1) / self.window
target_components_df = target_components_df.iloc[-prediction_size:]
return target_components_df
def _forecast(self, df: pd.DataFrame, prediction_size: int) -> np.ndarray:
"""Make autoregressive forecasts on a wide dataframe."""
self._validate_context(df=df, prediction_size=prediction_size)
expected_length = prediction_size + self.context_size
history = df.loc[:, pd.IndexSlice[:, "target"]].values
history = history[-expected_length:-prediction_size]
if np.any(np.isnan(history)):
raise ValueError("There are NaNs in a forecast context, forecast method requires context to be filled!")
num_segments = history.shape[1]
res = np.append(history, np.zeros((prediction_size, num_segments)), axis=0)
for i in range(self.context_size, len(res)):
res[i] = res[i - self.context_size : i : self.seasonality].mean(axis=0)
y_pred = res[-prediction_size:]
return y_pred
[docs] def forecast(self, ts: TSDataset, prediction_size: int, return_components: bool = False) -> TSDataset:
"""Make autoregressive forecasts.
Parameters
----------
ts:
Dataset with features
prediction_size:
Number of last timestamps to leave after making prediction.
Previous timestamps will be used as a context.
return_components:
If True additionally returns forecast components
Returns
-------
:
Dataset with predictions
Raises
------
NotImplementedError:
if return_components mode is used
ValueError:
if context isn't big enough
ValueError:
if forecast context contains NaNs
"""
df = ts.to_pandas()
y_pred = self._forecast(df=df, prediction_size=prediction_size)
ts.df = ts.df.iloc[-prediction_size:]
ts.df.loc[:, pd.IndexSlice[:, "target"]] = y_pred
if return_components:
# We use predicted targets as lags in autoregressive style
df.loc[df.index[-prediction_size:], pd.IndexSlice[:, "target"]] = y_pred
target_components_df = self._predict_components(df=df, prediction_size=prediction_size)
ts.add_target_components(target_components_df=target_components_df)
return ts
def _predict(self, df: pd.DataFrame, prediction_size: int) -> np.ndarray:
"""Make predictions on a wide dataframe using true values as autoregression context."""
self._validate_context(df=df, prediction_size=prediction_size)
expected_length = prediction_size + self.context_size
context = df.loc[:, pd.IndexSlice[:, "target"]].values
context = context[-expected_length:]
if np.any(np.isnan(context)):
raise ValueError("There are NaNs in a target column, predict method requires target to be filled!")
num_segments = context.shape[1]
res = np.zeros((prediction_size, num_segments))
for res_idx, context_idx in enumerate(range(self.context_size, len(context))):
res[res_idx] = context[context_idx - self.context_size : context_idx : self.seasonality].mean(axis=0)
y_pred = res[-prediction_size:]
return y_pred
[docs] def predict(self, ts: TSDataset, prediction_size: int, return_components: bool = False) -> TSDataset:
"""Make predictions using true values as autoregression context (teacher forcing).
Parameters
----------
ts:
Dataset with features
prediction_size:
Number of last timestamps to leave after making prediction.
Previous timestamps will be used as a context.
return_components:
If True additionally returns prediction components
Returns
-------
:
Dataset with predictions
Raises
------
NotImplementedError:
if return_components mode is used
ValueError:
if context isn't big enough
ValueError:
if forecast context contains NaNs
"""
df = ts.to_pandas()
y_pred = self._predict(df=df, prediction_size=prediction_size)
ts.df = ts.df.iloc[-prediction_size:]
ts.df.loc[:, pd.IndexSlice[:, "target"]] = y_pred
if return_components:
# We use true targets as lags
target_components_df = self._predict_components(df=df, prediction_size=prediction_size)
ts.add_target_components(target_components_df=target_components_df)
return ts
[docs] def params_to_tune(self) -> Dict[str, BaseDistribution]:
"""Get default grid for tuning hyperparameters.
This grid tunes ``window`` parameter. Other parameters are expected to be set by the user.
Returns
-------
:
Grid to tune.
"""
return {"window": IntDistribution(low=1, high=10)}
__all__ = ["SeasonalMovingAverageModel"]