%% Example-Appendix I.1: The 10-dimensional synthetic system with non-diagonal system matrix
clear;clc;close all;
addpath (genpath('tools')) 
rng(100, 'twister');
dim=10;ts=1;
R=0.81*eye(dim);
Q=R;
R = chol(R)';Q = chol(Q)';
xi = ones(dim,1);rr = 1;
iv = xi + rr * randn(dim,1);  

A = zeros(dim, dim);
B = zeros(dim, dim);
C = zeros(dim, dim);
D = zeros(dim, dim);
colsa = randperm(dim);
colsb = randperm(dim);
colsc = randperm(dim);
colsd = randperm(dim);

for row = 1:dim
    A(row, colsa(row)) = 0.8;
    B(row, colsb(row)) = 2;
    C(row, colsc(row)) = 2;
    D(row, colsd(row)) = 2;
end

n = size(A,1);
mm = size(C,1);
% Initial state 
T_total = 2100;
T_train = T_total*2/3;
T_test = T_total - T_train;   
% Training dataset
X_train = zeros(dim, T_train);
Y_train = zeros(dim, T_train);
for i = 1:T_train
    U_train(:,i) = 2 * rand(dim, 1); % Input signal
    if i == 1
        X_train(:,i) = A * iv + B * U_train(:,i) + R * randn(n,1);
    else
        X_train(:,i) = A * X_train(:,i-1) + B * U_train(:,i) + R * randn(n,1);
    end
        Y_train(:,i) = C * X_train(:,i) + D * U_train(:,i) + Q * randn(mm,1);
end
% Testing dataset
X_test =zeros(dim, T_test);
Y_test = zeros(dim, T_test);
U_test = zeros(dim, T_test);
for i = 1:T_test
    U_test(:,i) = 5 * rand(dim, 1);  
    if i == 1
        X_test(:,i) = A * iv + B * U_test(:,i) + R * randn(n,1);
    else
        X_test(:,i) = A * X_test(:,i-1) + B * U_test(:,i) + R * randn(n,1);
    end
    Y_test(:,i) = C * X_test(:,i) + D * U_test(:,i) + Q * randn(mm,1);
end

threshold=0.005;

%% LSM_HK
n = dim; mm = dim; p = dim;
T_hokalman = 2 * n; % Time length for Hankel matrix
Nb = size(U_train, 2); % Trajectory length
tic
[A_hokalman, B_hokalman, C_hokalman, D_hokalman] = LSM_HK(U_train, Y_train, Nb, n, mm, p, T_hokalman);
toc
A_hokalman(abs(A_hokalman) < threshold) = 0;B_hokalman(abs(B_hokalman) < threshold) = 0;
C_hokalman(abs(C_hokalman) < threshold) = 0;D_hokalman(abs(D_hokalman) < threshold) = 0;
% Prediction on test data
T_test = size(Y_test, 2);
[X_hokalman, Y_hokalman] = state_space_init_predict(A_hokalman, B_hokalman, C_hokalman, D_hokalman, U_test, Y_test, T_test);
norm_relative_errors_hokalman = calculate_relative_errors(Y_test, Y_hokalman);
average_norm_relative_error_hokalman = mean(norm_relative_errors_hokalman);

%% Our method
n = dim; mm = dim; p = dim; Iter_Max = 600;
tic
[A_our, B_our, C_our, D_our, R_our, Q_our] = Our_Method(Y_train, U_train, n, mm, p, T_train, Iter_Max, xi, rr,ts);
toc
%mean relative error
A_our(abs(A_our) < threshold) = 0;B_our(abs(B_our) < threshold) = 0;
C_our(abs(C_our) < threshold) = 0;D_our(abs(D_our) < threshold) = 0;
[X_our, Y_our] = state_space_init_predict(A_our, B_our, C_our, D_our, U_test, Y_test, T_test);
norm_relative_errors_our = calculate_relative_errors(Y_test, Y_our);
average_norm_relative_error_our = mean(norm_relative_errors_our);

%% MLE
tic
[A_mle, B_mle, C_mle, D_mle, R_mle, Q_mle] = MLE(Y_train, U_train, n, mm, p, T_train, Iter_Max, xi, rr,ts);
toc
A_mle(abs(A_mle) < threshold) = 0;B_mle(abs(B_mle) < threshold) = 0;
C_mle(abs(C_mle) < threshold) = 0;D_mle(abs(D_mle) < threshold) = 0;
% mean relative error
[X_mle, Y_mle] = state_space_init_predict(A_mle, B_mle, C_mle, D_mle, U_test, Y_test, T_test);
norm_relative_errors_mle = calculate_relative_errors(Y_test, Y_mle);
average_norm_relative_error_mle = mean(norm_relative_errors_mle);
%% N4SID 
Ts = 1;
data = iddata(Y_train', U_train', Ts);
tic
n4sid_sys = n4sid(data, dim);
toc
A_n4sid=n4sid_sys.A;B_n4sid=n4sid_sys.B;C_n4sid=n4sid_sys.C;D_n4sid=n4sid_sys.D;
A_n4sid(abs(A_n4sid) < threshold) = 0;B_n4sid(abs(B_n4sid) < threshold) = 0;
C_n4sid(abs(C_n4sid) < threshold) = 0;D_n4sid(abs(D_n4sid) < threshold) = 0;
% mean relative error
[X_n4sid_sys, Y_n4sid_sys] = state_space_init_predict(n4sid_sys.A, n4sid_sys.B, n4sid_sys.C, n4sid_sys.D, U_test, Y_test, T_test);
norm_relative_errors_n4sid = calculate_relative_errors(Y_test, Y_n4sid_sys);
average_norm_relative_error_n4sid = mean(norm_relative_errors_n4sid);
%% LSM_PEM
data = iddata(Y_train', U_train', 1);
A_in = eye(dim); B_in = eye(dim); C_in = eye(dim); D_in = eye(dim); K_in = eye(dim); x0 = ones(dim,1);
sys_in = idss(A_in, B_in, C_in, D_in, K_in, x0);
tic
pem_sys = pem(data, sys_in, 'Focus', 'prediction');
toc
A_pem=pem_sys.A;B_pem=pem_sys.B;C_pem=pem_sys.C;D_pem=pem_sys.D;
A_pem(abs(A_pem) < threshold) = 0;B_pem(abs(B_pem) < threshold) = 0;
C_pem(abs(C_pem) < threshold) = 0;D_pem(abs(D_pem) < threshold) = 0;
% mean relative error
[X_pem, Y_pem] = state_space_init_predict(pem_sys.A, pem_sys.B, pem_sys.C, pem_sys.D, U_test, Y_test, T_test);
norm_relative_errors_pem = calculate_relative_errors(Y_test, Y_pem);
average_norm_relative_error_pem = mean(norm_relative_errors_pem);
%% Display results
disp('Average Norm Relative Error (Our Method): ' + string(average_norm_relative_error_our));
disp('Average Norm Relative Error (MLE): ' + string(average_norm_relative_error_mle));
disp('Average Norm Relative Error (N4SID): ' + string(average_norm_relative_error_n4sid));
disp('Average Norm Relative Error (LSM_PEM): ' + string(average_norm_relative_error_pem));
disp('Average Norm Relative Error (LSM_HK): ' + string(average_norm_relative_error_hokalman));




