function res = test_linearSysDT_conform_check_02_knownUncertainty
% test_linearSysDT_conform_check_02_knownUncertainty - unit test function 
%   for conformance synthesis of linear discrete-time systems according to 
%   [1].
%
%   This is a sanity check. Test cases with known uncertainty are
%   produced. The conformnace check should return conformance and return 
%   false when the known uncertainties are reduced.
%
% Syntax:
%    res = test_linearSysDT_conform_check_02_knownUncertainty
%
% Inputs:
%    -
%
% Outputs:
%    res - true/false
%
% References:
%    [1] M. Althoff. Checking and Establishing Reachset Conformance in
%        CORA 2023. In Proc. of the 10th International Workshop on 
%        Applied Verification of Continuous and Hybrid Systems, 2023.
%    [2] Liu et al., "Guarantees for Real Robotic Systems: Unifying Formal
%        Controller Synthesis and Reachset-Conformant Identification", 202x.

% Authors:       Matthias Althoff
% Written:       12-July-2023
% Last update:   ---
% Last revision: ---

% ------------------------------ BEGIN CODE -------------------------------

% initialize partial results
resPartial = [];

%% System dynamics
% create pedestrian model (see Lecture "Formal Methods for Cyber-Physical
% Systems - Conformance Checking")
% sample time
dt = 0.4;
% discrete time system matrix from continuous system matrix Ac
Ac = [0 0 1 0; 0 0 0 1; 0 0 0 0; 0 0 0 0];
A = expm(Ac*dt);
B = [];
C = [1 0 0 0; 0 1 0 0];
D = [];

% instantiate linear discrete-time dynamics
pedestrian = linearSysDT(A,B,[],C,D,dt);
params.tFinal = 2;

% the system has known uncertainties
W = zonotope(rand(4,5)); % disturbance 
V = zonotope(rand(2,3)); % measurement uncertainty 
R0 = zonotope(rand(4,1)); % initial state uncertainty; initial state uncertainty has no effect for linear systems

% options to weight cost function of different time steps
maxNrOfTimeSteps = ceil(params.tFinal/dt); % maximum number of timeSteps
options.w = ones(maxNrOfTimeSteps+1,1);

%% Create test cases
% because the system has no imaginary eigenvalues, the extreme cases
% suffice to reproduce the actual uncertainties
% combine zonotopes for R0, W, and V in a single zonotope
S = cartProd(W, V);
% compute vertices of combined zonotope
Vmat = vertices(S);
% set initial state
params.R0 = R0;
% init test cases
nrOfTests = size(Vmat,2);
synthesizedTests = cell(nrOfTests,1);
% loop over each vertex
for i = 1:nrOfTests
    % overwrite parameters
    params.W = zonotope(Vmat(1:4,i)); % disturbance 
    params.V = zonotope(Vmat(5:6,i)); % measurement uncertainty  
    % simulate system
    simOpt.points = 1;
    simRes = simulateRandom(pedestrian,params,simOpt);
    % construct input vector
    uVec = zeros(maxNrOfTimeSteps+1,1);
    % save in test case
    synthesizedTests{i} = testCase(simRes.y{1}, uVec, simRes.x{1}, dt);
end

    
%% Conformance check (result should be true)
options.confAlg = 'dyn';
options.zonotopeOrder = 200;
options.norm = 'interval';
params.testSuite = synthesizedTests;
% store uncertainty
params.W = W; % disturbance 
params.V = enlarge(V,1.001); % measurement uncertainty enlarges by 0.1%
params.R0conf = R0 - center(R0); % initial state uncertainty for conformance relative to x0
resPartial(end+1) = conform(pedestrian,'check',params,options); 

%% Reduce measurement uncertainty to 99.9% (result should be false)
params.V = enlarge(V,0.999);
resPartial(end+1) = ~conform(pedestrian,'check',params,options);

% overall result
res = all(resPartial);

end
        
% ------------------------------ END OF CODE ------------------------------
