-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathWorkhorseFunctions.py
92 lines (75 loc) · 3.52 KB
/
WorkhorseFunctions.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
"""
Разработанный Адриелу Ванг от ДанСтат Консульти́рования
"""
import torch
class WorkhorseFunctions:
@staticmethod
def ols_estimator_torch(X, y):
y = y.view(-1, y.shape[-1]) # Reshape y to have the right dimensions
XtX = X.t().mm(X)
Xty = X.t().mm(y)
beta_hat = torch.linalg.solve(XtX, Xty)
return beta_hat
@staticmethod
def create_input_output_pairs(data, n_lags):
X, y = [], []
for i in range(n_lags, len(data)):
X.append(data[i - n_lags:i].flatten())
y.append(data[i])
return torch.stack(X), torch.stack(y)
class TimeSeriesWorkhorse:
# Initialize AR and MA parameters using OLS estimation
def initialize_params_torch(y, p, q):
# AR part
X_ar = torch.zeros((len(y) - p, p), dtype=torch.float64)
for t in range(p, len(y)):
for i in range(p):
X_ar[t - p, i] = y[t - i - 1]
ar_coeffs = WorkhorseFunctions.ols_estimator_torch(X_ar, y[p:].view(-1, 1))
# Compute the residuals
residuals = y[p:] - X_ar.mm(ar_coeffs).view(-1)
# MA part
X_ma = torch.zeros((len(residuals) - q, q), dtype=torch.float64)
for t in range(q, len(residuals)):
for j in range(q):
X_ma[t - q, j] = residuals[t - j - 1]
ma_coeffs = WorkhorseFunctions.ols_estimator_torch(X_ma, residuals[q:].view(-1, 1))
return ar_coeffs.view(-1), ma_coeffs.view(-1)
# Compute the negative log-likelihood and gradients for the ARIMA model
def negative_log_likelihood_torch(params, y, p, d, q):
ar_params = params[:p]
ma_params = params[p:p + q]
intercept = params[-1]
y_hat = torch.zeros_like(y)
for t in range(p, len(y)):
ar_term = sum(ar_params[i] * y[t - i - 1] for i in range(p))
ma_term = sum(ma_params[j] * (y[t - j - 1] - intercept) for j in range(q))
y_hat[t] = intercept + ar_term + ma_term
residuals = y - y_hat
sigma2 = torch.sum(residuals ** 2) / len(residuals)
log_likelihood = -0.5 * (len(residuals) * torch.log(2 * torch.acos(torch.zeros(1)).item() * 2 * sigma2) + torch.sum(residuals ** 2) / sigma2)
gradients = torch.zeros_like(params)
for t in range(p, len(y)):
for i in range(p):
gradients[i] += residuals[t] * y[t - i - 1]
for j in range(q):
gradients[p + j] += residuals[t] *(residuals[t - j - 1] - intercept)
gradients[-1] += residuals[t]
gradients /= sigma2
return -log_likelihood, -gradients
def arima_estimator_torch(y, p, d, q, learning_rate=0.01, n_iterations=500):
if not isinstance(y, torch.Tensor):
y = torch.tensor(y, dtype=torch.float64)
if d > 0:
y = torch.diff(y, n=d)
# Initialize the AR, MA, and intercept parameters using OLS
ar_coeffs, ma_coeffs = TimeSeriesWorkhorse.initialize_params_torch(y, p, q)
params = torch.cat((ar_coeffs, ma_coeffs, torch.zeros(1, dtype=torch.float64)), dim=0)
# Optimize the negative log-likelihood using custom SGD
for i in range(n_iterations):
neg_loglik, neg_grads = TimeSeriesWorkhorse.negative_log_likelihood_torch(params, y, p, d, q)
params -= learning_rate * neg_grads
ar_coeffs = params[:p].flip(0)
ma_coeffs = params[p:p + q]
intercept = params[-1]
return ar_coeffs, ma_coeffs, intercept