Neural DDEยค
Warning
This library only supports constant lag DDEs. Therefore we are unable to model time and state dependent DDEs.
This examples trains a Neural DDE to reproduce a simple dataset of a delay logistic equation. In this example, the backward pass is computed with the adjoint method.
import time
import matplotlib.pyplot as plt
import torch
import torch.nn as nn
from torch.utils.data import DataLoader, Dataset
from torchdde import integrate, Euler
from torchvision.ops import MLP
device = torch.device("cuda" if torch.cuda.is_available() else "cpu")
Recalling that a Neural DDE is defined as
\[\frac{dy}{dt} = f_{\theta}(t, y(t), y(t-\tau_1), \dots, y(t-\tau_{n})), \quad y(t<0) = \psi(t)\]
then here we're now about to define \(f_{\theta}\) that appears on that right hand side on the equation above
class NDDE(nn.Module):
def __init__(
self,
delays,
in_size,
out_size,
width_size,
depth,
):
super().__init__()
self.in_dim = in_size * (1 + len(delays))
self.delays = torch.nn.Parameter(delays)
self.mlp = MLP(
self.in_dim,
hidden_channels=depth * [width_size] + [out_size],
)
def forward(self, t, z, args, *, history):
# `history` corresponds to the list of
# delayed states defined in your DDE
# i.e. here history=[y(t-tau1), ..., y(t-taun)]
return self.mlp(torch.cat([z, *history], dim=-1))
We generate the toy dataset of the delayed logistic equation (Equation 2.1).
def get_data(y0, ts, tau=torch.tensor([1.0])):
def f(t, y, args, history):
return y * (1 - history[0])
history_function = lambda t: torch.unsqueeze(y0, dim=1)
ys = integrate(f, Euler(), ts[0], ts[-1], ts, history_function, args=None, dt0=ts[1]-ts[0], delays=tau)
return ys
class MyDataset(Dataset):
def __init__(self, ys):
self.ys = ys
def __getitem__(self, index):
return self.ys[index]
def __len__(self):
return self.ys.shape[0]
Main entry point. Try running main()
.
def main(
dataset_size=128,
batch_size=128,
lr=0.0005,
max_epoch=1000,
width_size=32,
depth=2,
seed=5678,
plot=True,
print_every=5,
device=device,
):
torch.manual_seed(seed)
ts = torch.linspace(0, 10, 101)
y0_min, y0_max = 2.0, 3.0
y0 = (y0_min - y0_max) * torch.rand((dataset_size,)) + y0_max
ys = get_data(y0, ts)
ts, ys = ts.to(device), ys.to(device)
delay_min, delay_max = 0.7, 1.3
value = (delay_max - delay_min) * torch.rand((1,)) + delay_min
tau = torch.tensor([value], device=device)
tau = tau.to(device)
state_dim = ys.shape[-1]
model = NDDE(tau, state_dim, state_dim, width_size, depth)
model = model.to(device)
dataset = MyDataset(ys)
train_loader = DataLoader(dataset, batch_size=batch_size, shuffle=True)
# Training loop like normal.
model.train()
loss_fn = torch.nn.MSELoss()
optimizer = torch.optim.Adam(model.parameters(), lr=lr)
for epoch in range(max_epoch):
for step, data in enumerate(train_loader):
t = time.time()
optimizer.zero_grad()
data = data.to(device)
history_fn = lambda t: data[:, 0]
ys_pred = integrate(model, Euler(), ts[0], ts[-1], ts, history_fn, args=None, dt0=ts[1]-ts[0], delays=tau)
loss = loss_fn(ys_pred, data)
loss.backward()
optimizer.step()
if (epoch % print_every) == 0 or epoch == max_epoch - 1:
print(
"Epoch : {}, Step {}/{}, Loss : {:.3e}, Tau {}, Time {}".format(
epoch,
step + 1,
len(train_loader),
loss.item(),
[d.item() for d in model.delays],
time.time() - t,
)
)
if plot:
plt.plot(ts.cpu(), data[0].cpu(), c="dodgerblue", label="Real")
history_values = data[0, 0][..., None]
history_fn = lambda t: history_values
ys_pred = integrate(model, Euler(), ts[0], ts[-1], ts, history_fn, args=None, dt0=ts[1]-ts[0], delays=tau)
plt.plot(
ts.cpu(),
ys_pred[0].cpu().detach(),
"--",
c="crimson",
label="Model",
)
plt.legend()
plt.savefig("neural_dde.png")
plt.show()
plt.close()
return ts, ys, model
ts, ys, model = main()