function [params, results] = conform(sys,params,options,varargin)
% conform - performs reachset conformance by computing the reachable 
%         of a system and checking whether all measurements are included
%
% Syntax:
%    [params, results] = conform(sys,params,options)
%    [params, results] = conform(sys,params,options,type)
%
% Inputs:
%    sys - contDynamics object
%    params - parameter defining the conformance problem
%    options - options for the conformance checking
%    type - (optional) "RRT", "graySim", "graySeq", "grayLS" or "white",
%           can be omitted for "white"
%
% Outputs:
%    params - struct with conformant parameters
%    results - struct with the optimization results (dependent on type)
%       .R - reachSet object (only time steps for which measurments exist)
%       .simRes - states of the rapidly exploring random tree
%       .unifiedOutputs - unified test cases (only deviation to nominal 
%           solution)
%       .sys - updated system
%       .p - estimated parameters
%       .fval - final optimization cost
%
% References:
%    [1] M. Althoff and J. M. Dolan. Reachability computation of low-order 
%        models for the safety verification of high-order road vehicle 
%        models. In Proc. of the American Control Conference, 
%        page 3559–3566, 2012.
%    [2] L. Luetzow and M. Althoff, "Reachset-conformant System
%        Identification," arXiv, 2024. 
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: none

% Authors:       Matthias Althoff, Laura Luetzow
% Written:       15-June-2023
% Last update:   21-March-2024 (LL, change name from confSynth to conform)
% Last revision: ---

% ------------------------------ BEGIN CODE -------------------------------

% set default type
type = setDefaultValues({"white"},varargin);

% select conformance algorithm
switch type
    case "RRT"
        % synthesize conformance using RRTs, see [1]    
        [params, R, simRes] = conform_RRT(sys, params, options);
        results.R = R;
        results.simRes = simRes;

    otherwise

        if contains(type,"black") % blackGP, blackCGP
            % approximate the system dynamics
            sys = conform_black(params, options, type);
            results.sys = sys;
            options = rmfield(options, 'approx');
            params = rmfield(params, ["testSuite_train","testSuite_val"]);
        end
    
        % conformance synthesis for dedicated system dynamics
        [params,options] = validateOptions(sys, params, options);
        [sys_upd, params, options] = aux_updateSys(sys,params,options);
    
        % Identify conformant parameters
        if type == "white" || contains(type,"black")
            [params, fval, p_opt,  union_y_a] = conform_white(sys_upd, params, options);
            results.sys = sys;
            results.unifiedOutputs = union_y_a;
    
        elseif contains(type,"gray") % "graySim", "graySeq", "grayLS"
            [params, fval, p_opt, sys_opt] = conform_gray(sys_upd, params, options, type);
            results.sys = sys_opt;
        end
    
        params = aux_updateUWV(params);
        results.fval = fval;
        results.p = p_opt;

end

end


% Auxiliary functions -----------------------------------------------------

function [sys, params, options] = aux_updateSys(sys, params, options)
% preprocess sys, parameters and options

% augment input set with the additive uncertainty sets W and V if given
if isfield(params, 'W')
    params.U = cartProd(params.U, params.W);
    % update dynamical system
    sys = augment_u_with_w(sys);
end
if isfield(params, 'V')
    params.U = cartProd(params.U, params.V);
    % update dynamical system
    sys = augment_u_with_v(sys);
end

% add the dimensions of the optimization variables to options
options.cs.n_a = size(params.U.generators,2) + size(params.R0.generators,2);
if isa(sys, 'linearSysDT') || isa(sys, 'linearARX')
    % c can be computed with linear programming
    options.cs.n_c = size(params.U.generators,1) + size(params.R0.generators,1);
else
    % c is not estimated in the linear program
    options.cs.n_c = 0;
end

% adapt testSuite to the system dynamics (augment nominal input, update
% initial state)
params.testSuite = aux_preprocessTestSuite(sys, params.testSuite);

end

function params = aux_updateUWV(params)

% split U in the disturbance sets W and V if they were initially given
if isfield(params, "V")
    dim_u = 1:dim(params.U)-dim(params.V);
    dim_v = dim(params.U)-dim(params.V)+1:dim(params.U);
    params.V = project(params.U, dim_v);
    params.U = project(params.U, dim_u);
end
if isfield(params, "W")
    dim_u = 1:dim(params.U)-dim(params.W);
    dim_w = dim(params.U)-dim(params.W)+1:dim(params.U);
    params.W = project(params.U, dim_w);
    params.U = project(params.U, dim_u);
end

end

function testSuite = aux_preprocessTestSuite(sys, testSuite)
% preprocessTestSuite - preprocess testSuite, i.e., 
%   - augment nominal inputs with zeros to match input dimension of sys
%   - unify test cases for linear systems, 
%   - adapt the initial state to the given system dynamics

% Augment nominal input with zeros
for m = 1:length(testSuite)
    diff_u = sys.nrOfInputs - size(testSuite{m}.u,2);
    if diff_u > 0
        u_in = cat(2, testSuite{m}.u, zeros(size(testSuite{m}.u,1), diff_u, size(testSuite{m}.u,3)));
        testSuite{m} = testCase(testSuite{m}.y, u_in, testSuite{m}.initialState, sys.dt, class(sys));
    end
end

% Combine test cases for linear systems
if (isa(sys, 'linearARX') || isa(sys, 'linearSysDT')) ...
        && length(testSuite) > 1
    testSuite_m = combineTestCases(testSuite{1},testSuite{2});
    for m=3:length(testSuite)
        testSuite_m = combineTestCases(testSuite_m,testSuite{m});
    end
    testSuite = {};
    testSuite{1} = testSuite_m;
end

% Adaption initial state to state-space models
if ~isa(sys, 'linearARX') && ~isa(sys, 'nonlinearARX') 
    if sys.dim == size(testSuite{1}.initialState,1)
        % all good
        return
    end
    throw(CORAerror('CORA:notSupported',['Transformation of test cases created by input-output-models ' ...
        'to state-space models is not implemented, yet.']))
end

% Adaption initial state to input-output models
if contains(testSuite{1}.name, ["ARX", "ARX"]) && ...
        sys.dim*sys.nrOfOutputs == size(testSuite{1}.initialState,1)
    % testSuite was generated by an input-output model with the same
    % model dimension
    % -> initial state should be set correctly
    return
end

% test suite was generated by a state-space model or an input-output model
% with different model dimension
n_m = length(testSuite);
testSuite_new = {};
for m=1:n_m
    testSuite_new = [testSuite_new; setInitialStateToMeas(testSuite{m},sys.dim)];
end
testSuite = testSuite_new;

end

% ------------------------------ END OF CODE ------------------------------
