function res = test_polytope_vertices
% test_polytope_vertices - unit test function of vertices
%
% Syntax:
%    res = test_polytope_vertices
%
% Inputs:
%    -
%
% Outputs:
%    res - true/false
%
% Other m-files required: none
% Subfunctions: none
% MAT-files required: none
%
% See also: none

% Authors:       Mark Wetzlinger
% Written:       29-November-2022
% Last update:   24-November-2023 (MW, check errors for unbounded cases)
% Last revision: ---

% ------------------------------ BEGIN CODE -------------------------------

res = true(0);

% 1D, empty, only inequalities
P = polytope([2; 1; -4],[5; 0.5; -3]);
V = vertices(P);
res(end+1,1) = isempty(V);

% 1D, empty, inequalities and equalities
P = polytope([1;-4],[4;-2],5,100);
V = vertices(P);
res(end+1,1) = isempty(V);

% 1D, bounded
P = polytope([1; -2],[1.25; 1]);
V = vertices(P);
V_ = [-0.5, 1.25];
res(end+1,1) = all(withinTol(V,V_));

% 1D, bounded, non-minimal representation
P = polytope([2; 1; -1; -2],[1; 2; 5; 2]);
V = vertices(P);
V_ = [-1 0.5];
res(end+1,1) = all(withinTol(V,V_));

% 1D, unbounded
P = polytope(1, 0);
V = vertices(P);
V_ = [-Inf, 0];
res(end+1,1) = all(V == V_);

% 1D, unbounded
P = polytope(3,6);
V = vertices(P);
V_ = [-Inf, 2];
res(end+1,1) = all(V == V_);

% 1D, degenerate, only inequalities
P = polytope([2; -1],[4; -2]);
V = vertices(P);
V_ = 2;
res(end+1,1) = all(withinTol(V,V_));

% 1D, degenerate, equality
P = polytope([],[],1,2);
V = vertices(P);
V_ = 2;
res(end+1,1) = all(withinTol(V,V_));


% 2D, bounded
P = polytope([2 1; 2 -2; -2 -2; -1 2],[2; 2; 2; 2]);
V = vertices(P);
V_comb = vertices(P,'comb');
V_ = [0.4 1.2; 1 0; 0 -1; -4/3 1/3]';
res(end+1,1) = compareMatrices(V,V_) && compareMatrices(V,V_comb,1e-14);

% 2D, vertex instantiation
V_ = [3 0; 2 2; -1 3; -2 0; 0 -1]';
P = polytope(V_);
V = vertices(P);
V_comb = vertices(P,'comb');
res(end+1,1) = compareMatrices(V,V_,1e-14) && compareMatrices(V,V_comb,1e-14);

% 2D, bounded, degenerate (single point)
P = polytope([1 1; 1 -1; -1 0],zeros(3,1));
V = vertices(P);
V_comb = vertices(P,'comb');
V_ = [0;0];
res(end+1,1) = compareMatrices(V,V_,1e-14) && compareMatrices(V,V_comb,1e-14);

% 2D, bounded, degenerate (line)
P = polytope([1 1; 1 -1; -1 -1; -1 1],[1; 0; 1; 0]);
V = vertices(P);
V_comb = vertices(P,'comb');
V_ = [0.5 0.5; -0.5 -0.5]';
res(end+1,1) = compareMatrices(V,V_,1e-14) && compareMatrices(V,V_comb,1e-14);

% 2D, unbounded (should throw an error!)
P = polytope([1 0],1);
try
    V = vertices(P);
    res(end+1,1) = false;
end

% 2D, unbounded (should throw an error!)
P = polytope([1 0; -1 0],[1;1]);
try
    V = vertices(P);
    res(end+1,1) = false;
end
[M,~,~] = svd(rand(2));
P_ = M*P;
try
    V = vertices(P_);
    res(end+1,1) = false;
end

% 2D, unbounded (should throw an error!)
P = polytope([1 0; -1 0; 0 -1],[1;1;1]);
try
    V = vertices(P);
    res(end+1,1) = false;
end

% 2D, unbounded but not axis-aligned (should throw an error!)
P = polytope([1 -0.1; 0.1 -1; -0.1 -1; -1 -0.1],ones(4,1));
try
    V = vertices(P);
    res(end+1,1) = false;
end


% 3D, degenerate (2D simplex)
A = [-1 0 0; 0 -1 0; 1 1 0]; b = [0;0;2];
Ae = [0 0 1]; be = 0;
P = polytope(A,b,Ae,be);
V = vertices(P);
V_ = [2 0 0; 0 2 0; 0 0 0]';
res(end+1,1) = compareMatrices(V,V_,1e-14);

% 3D, unit box
n = 3;
P = polytope([eye(n); -eye(n)],[ones(2*n,1)]);
V = vertices(P);
V_ = [1 1 1; 1 1 -1; 1 -1 1; -1 1 1; 1 -1 -1; -1 1 -1; -1 -1 1; -1 -1 -1]';
res(end+1,1) = compareMatrices(V,V_);

% 3D, degenerate unit box: square 
n = 3;
P = polytope([eye(n); -eye(n)],[1; 1; 0; 1; 1; 0]);
V = vertices(P);
V_comb = vertices(P,'comb');
V_ = [1 1 0; -1 1 0; -1 -1 0; 1 -1 0]';
res(end+1,1) = compareMatrices(V,V_,1e-14) && compareMatrices(V,V_comb,1e-14);


% 4D, degenerate (unit square)
A = [eye(2) zeros(2); -eye(2) zeros(2)]; b = [ones(4,1)];
Ae = [0 0 1 0; 0 0 0 1]; be = [0;0];
P = polytope(A,b,Ae,be);
V = vertices(P);
V_ = [1 1 0 0; 1 -1 0 0; -1 1 0 0; -1 -1 0 0]';
res(end+1,1) = compareMatrices(V,V_,1e-14);

% 4D, degenerate (rotated unit square)
A = [eye(2) zeros(2); -eye(2) zeros(2)]; b = [ones(4,1)];
Ae = [0 0 1 0; 0 0 0 1]; be = [0;0];
P = polytope(A,b,Ae,be);
M = [1 3 -2 4; 3 -2 4 -1; 3 -2 -1 3; 4 -3 -2 1];
V = vertices(M*P);
res(end+1,1) = compareMatrices(V,M*V_,1e-14);


% 7D, degenerate (subspace computation required)
A = [0       1  0   0  0  0   0;
     0      -1  0   0  0  0   0;
     0       0  0   1  0  0   0;
     0       0  0  -1  0  0   0;
     0       0  0   0  1  0   0;
     0       0  0   0 -1  0   0;
     0       0  0   0  0  0   1;
     0       0  0   0  0  0  -1;
     sqrt(2) 0 -0.5 0  0  0.5 0;
     sqrt(2) 0  0.5 0  0 -0.5 0;
     0       0  0   0  0  1   0;
     0       0 -1   0  0  0   0;
    -1       0  0   0  0  0   0];
b = [0;0;1;-1;0;0;0;0;sqrt(2);sqrt(2);0.26;-0.25;-1];
P = polytope(A,b);
V = vertices(P);
V_ = [1 0 0.25 1 0 0.25 0; 1 0 0.26 1 0 0.26 0]';
res(end+1,1) = compareMatrices(V,V_,1e-10);


% combine results
res = all(res);

% ------------------------------ END OF CODE ------------------------------
