--- title: "Introduction to gratis" author: "Bocong Zhao and Rob J Hyndman" output: rmarkdown::html_vignette vignette: > %\VignetteIndexEntry{Introduction to gratis} %\VignetteEngine{knitr::rmarkdown} %\VignetteDepends{feasts} %\VignetteEncoding{UTF-8} --- ```{r initial, echo = FALSE, cache = FALSE, results = 'hide'} library(knitr) opts_chunk$set( warning = TRUE, message = TRUE, echo = TRUE, cache = TRUE, fig.width = 7, fig.height = 4, fig.align = 'centre', comment = "#>" ) ``` # About gratis The gratis package generates synthetic time series data with diverse and controllable characteristics. It uses Gaussian mixture autoregressive (MAR) models to generate a wide range of non-Gaussian and nonlinear time series. The theory and methods are described in [Kang, Li and Hyndman (2020)](https://doi.org/10.1002/sam.11461). Synthetic time series data can be used to train or evaluate new algorithms for tasks such as time series forecasting, clustering and classification, with limited input of human effort or computational resources. The gratis package can generate data that mimics and expands real data sets, or which is more diverse than existing real data. Prof. Rob Hyndman also provided [a video tutorial available on YouTube]( https://www.youtube.com/watch?v=F3lWECtFa44). # Generate diverse time series ```{r setup} library(gratis) library(feasts) set.seed(5) ``` A MAR model is a mixture of $k$ Gaussian ARIMA$(p,d,0)(P,D,0)_m$ processes of the form $$ (1-B)^{d_i}(1-B^{m_i})^{D_i} (1-\phi_i(B))(1-\Phi_i(B)) y_t = c_i + \sigma_{i,t}\epsilon_t $$ with probability $\alpha_i$, where $B$ is the backshift operator, $m_i$ is the seasonal period, $\epsilon_t$ is a N(0,1) variate, and $\phi_i(B)$ and $\Phi_i(B)$ are polynomials in $B$ of order $p_i$ and $P_i$ respectively. The function `mar_model()` generates a MAR model with randomly selected parameters. The orders are uniformly sampled such that $p \in \{0,1,2,3\}$, $d \in \{0,1,2\}$, $P\in \{0,1,2\}$ and $D \in\{0,1\}$ (with the restriction that $d+D \le 2$). The parameters $\phi_{j,i}$ and $\Phi_{j,i}$ are uniformly sampled from the stationary parameter space, while the $\sigma_{i}$ values are uniformly sampled on $(1,5)$ and the mixture weights are uniformly sampled on $(0,1)$. The number of components is uniformly sampled on $\{1,2,3,4,5\}$. If required, each of these parameters can be specified by the user, rather than randomly selected. The resulting model object can be passed to `generate()` to return a `tsibble` of time series generated from the model. Alternatively, it can be passed to `simulate()` to return one time series using either the `ts` or `msts` class (depending on whether there is more than one seasonal period). Suppose we want to generate a random MAR model, and then generate 9 quarterly time series from it, each of length 5 years. ```{r marmodel} qmar <- mar_model(seasonal_periods = 4) qmar ``` This shows $k=`r ncol(qmar$ar)`$ components with weights `r sprintf("%.2f",qmar$weights)`. Now we can generate time series from this model. ```{r marplot} qmar %>% generate(nseries = 9, length = 20) %>% autoplot(value) ``` Each of these series comes from the same MAR model but with different stochastic inputs. Although the two ARIMA models are seasonal, the seasonality is too weak to been in the plots. # Generate multiple seasonal time series Time series can exhibit multiple seasonal patterns of different lengths, especially when series are observed at a high frequency such as daily or hourly data. Here is an example in which we generate 1 hourly time series of length 2 weeks. ```{r hmarmodel} hmar <- mar_model(seasonal_periods = c(24, 7*24)) hmar %>% generate(nseries = 1, length= 2*7*24) %>% autoplot(value) ``` This particular example shows strong time-of-day seasonality but no obvious day-of-week seasonality. In the next section we will see how to generate series with specific characteristics such as seasonality and trend. # Generate time series with targetted features The functions `generate_target()` and `simulate_target()` can efficiently generate time series with targetted features. These use a genetic algorithm to tune the MAR parameters until the distance between the target feature vector and the feature vector of the synthetic time series is as small as possible. As before, the `generate...` function returns a `tsibble` while the `simulate...` function returns a `ts` or `msts` object. Suppose we want to use generate a time series with the same level of trend and seasonality as the `USAccDeaths` data. First we create a function to measure the features we want to target. This time we will use `simulate()` rather than `generate()` so that the resulting time series has the same class as the `USAccDeaths` data. ```{r myfeatures} library(tsfeatures) my_features <- function(y) { c(stl_features(y)[c("trend", "seasonal_strength", "peak", "trough")]) } y <- simulate_target( length = length(USAccDeaths), seasonal_periods = frequency(USAccDeaths), feature_function = my_features, target = my_features(USAccDeaths) ) # Make new series same scale and frequency as USAccDeaths y <- ts(scale(y) * sd(USAccDeaths) + mean(USAccDeaths)) tsp(y) <- tsp(USAccDeaths) cbind(USAccDeaths, y) %>% autoplot() cbind(my_features(USAccDeaths), my_features(y)) ``` Next we will demonstrate the `generate_target()` function with target features specified by the spectral entropy and the first two autocorrelation coefficients. ```{r} library(dplyr) my_features <- function(y) { c(entropy(y), acf = acf(y, plot = FALSE)$acf[2:3, 1, 1]) } df <- generate_target( length = 60, feature_function = my_features, target = c(0.5, 0.9, 0.8) ) df %>% as_tibble() %>% group_by(key) %>% summarise(value = my_features(value), feature=c("entropy","acf1", "acf2"), .groups = "drop") df %>% autoplot(value) ``` # ARIMA and ETS models Just as `mar_model()` returns a MAR model, `arima_model()` and `ets_model()` will return ARIMA and ETS models. In all cases, elements will be selected randomly if the corresponding argument is omitted. For example, ```{r arimamodel} mod <- arima_model(frequency = 4) mod ``` This can then be passed to `generate()` or `simulate()` to obtain synthetic data from the model. The `simulate` methods for ARIMA and ETS models are actually from the forecast package rather than the gratis package.