% Federated Randomized SVD Algorithm Main Function
% Implements randomized SVD in a federated learning environment
% Inputs:
%   A_clients: Cell array of client data matrices, where each element is a client's local data matrix
%   params: Struct containing the following fields:
%     - k: Target rank (low-rank approximation dimension for SVD)
%     - p_over: Oversampling parameter, used to increase the stability of random projection
%     - rho: Diagonal decay factor for the ill-conditioned projection
%     - q: Number of power iterations
%     - enable_client_correction: Whether to enable client-side correction
%     - upload_corrected_sv: Whether to upload client-corrected singular values
% Outputs:
%   U_k: Global left singular vector matrix
%   S_k: Diagonal matrix of singular values
%   V_k: Right singular vector matrix
%   privacy_info: Analysis results of privacy protection effect

function [U_k, S_k, V_k, privacy_info] = federated_randomized_svd(A_clients, params)
    % Get the number of clients and data matrix dimensions
    num_clients = length(A_clients); % Number of clients
    [~, n] = size(A_clients{1}); % Number of columns in the data matrix
    d = params.k + params.p_over; % Target dimension of the random projection (target rank + oversampling)

    % Stage 1: Federated Range Exploration and Destructive Projection
    % Simplified Algorithm Comment:
    % An ill-conditioned projector, Omega_ill, is used to create low-dimensional sketches of client data.
    % This projection enhances privacy by introducing a decay factor rho.

    % 1. Server: Construct and distribute the ill-conditioned projector
    [U_omega, ~] = qr(randn(n, n), 0); % Generate n x n random orthogonal matrix U_omega
    [V_omega, ~] = qr(randn(d, d), 0); % Generate d x d random orthogonal matrix V_omega
    D_omega = diag(params.rho.^(0:d-1)); % Construct the ill-conditioned diagonal matrix with decay factor rho
    Omega_ill = U_omega(:, 1:d) * D_omega * V_omega'; % Construct the ill-conditioned projector

    %Omega_ill = eye(n,d);

    % 2. Clients: Compute local data sketches
    % Simplified Algorithm Comment:
    % Each client computes Y_i = A_i * Omega_ill. Y_i is a low-dimensional sketch of the local data.
    Y_clients = cell(num_clients, 1); % Initialize cell array for client sketches
    for i = 1:num_clients
        Y_clients{i} = A_clients{i} * Omega_ill; % Client i computes its local sketch
    end

    % 3. Power Iteration
    % Simplified Algorithm Comment:
    % Power iteration is used to better approximate the dominant subspace of A.
    % The global gradient G = sum(A_i' * Y_i) is aggregated on the server.
    for iter = 1:params.q
        % Clients compute local gradients
        G_clients = cell(num_clients, 1); % Initialize cell array for client gradients
        for i = 1:num_clients
            G_clients{i} = A_clients{i}' * Y_clients{i}; % Client i computes its local gradient
        end

        % Server aggregates the global gradient
        G = zeros(n, d); % Initialize global gradient matrix
        for i = 1:num_clients
            G = G + G_clients{i}; % Aggregate gradients from all clients
        end

        % Clients update their local sketches
        for i = 1:num_clients
            Y_clients{i} = A_clients{i} * G; % Update local sketch using the global gradient
        end
    end

    % 4. Clients: Local Orthogonalization
    % Simplified Algorithm Comment:
    % Each client performs QR decomposition on its Y_i to get an orthogonal basis Q_Y_i.
    Q_Y_clients = cell(num_clients, 1); % Initialize cell array for client orthogonal bases
    R_Y_clients = cell(num_clients, 1); % Initialize cell array for client upper triangular matrices
    for i = 1:num_clients
        [Q_Y_clients{i}, R_Y_clients{i}] = qr(Y_clients{i}, 0); % Client i performs QR decomposition
    end

    % 5. Server: Aggregation and Global Orthogonalization
    % Simplified Algorithm Comment:
    % The server stacks all R_Y_i, performs a QR decomposition to get a global orthogonal basis,
    % and then splits it back into blocks for each client.
    R_prime_Y = []; % Initialize stacked matrix
    for i = 1:num_clients
        R_prime_Y = [R_prime_Y; R_Y_clients{i}]; % Stack all client R_Y_i matrices
    end
    [Q_R, R_agg] = qr(R_prime_Y, 0); % Perform QR decomposition on the stacked matrix

    % Split Q_R into blocks for each client
    Q_R_clients = cell(num_clients, 1); % Initialize cell array for client Q_R blocks
    start_idx = 1;
    for i = 1:num_clients
        end_idx = start_idx + d - 1;
        Q_R_clients{i} = Q_R(start_idx:end_idx, :); % Assign Q_R block for client i
        start_idx = end_idx + 1;
    end

    % Stage 2: Federated Projection and Centralized SVD
    % Simplified Algorithm Comment:
    % Clients project their data, the server aggregates these projections into a small matrix B,
    % and then performs SVD on B to get the core SVD components.

    % 6. Clients: Compute local projected components
    B_prime_clients = cell(num_clients, 1); % Initialize cell array for client projected components
    for i = 1:num_clients
        B_prime_clients{i} = Q_Y_clients{i}' * A_clients{i}; % Client i computes its local projected component
    end

    % 7. Server: Construct and decompose the low-dimensional matrix
    B = zeros(d, n); % Initialize the low-dimensional matrix B
    for i = 1:num_clients
        B = B + Q_R_clients{i}' * B_prime_clients{i}; % Aggregate projected components from all clients
    end

    % Perform SVD on B
    [U_tilde_k, S_k, V_k] = svd(B, 'econ'); % Economy-size SVD
    U_tilde_k = U_tilde_k(:, 1:params.k); % Truncate to the top k left singular vectors
    S_k = S_k(1:params.k, 1:params.k); % Truncate to the top k singular values
    V_k = V_k(:, 1:params.k); % Truncate to the top k right singular vectors

    % Stage 3: Client-side SVD Correction
    % Simplified Algorithm Comment:
    % If client correction is enabled, clients compute their local shares of the left singular vectors.
    % Optionally, they can also correct the singular values using their local data.

    if params.enable_client_correction
        % 8. Clients: Compute local left singular vector segments
        U_k_clients = cell(num_clients, 1); % Initialize cell array for client left singular vectors
        for i = 1:num_clients
            U_k_clients{i} = Q_Y_clients{i} * Q_R_clients{i} * U_tilde_k; % Client i computes its local left singular vectors
        end

        % 9. Clients: Correct singular values using local full data
        if params.upload_corrected_sv
            S_k_clients = cell(num_clients, 1); % Initialize cell array for client singular values
            for i = 1:num_clients
                S_k_clients{i} = U_k_clients{i}' * A_clients{i} * V_k; % Client i computes its local singular value contribution
            end

            % 10. Server: Aggregate local singular value contributions
            S_k = zeros(params.k, params.k); % Initialize global singular value matrix
            for i = 1:num_clients
                S_k = S_k + S_k_clients{i}; % Aggregate singular value contributions from all clients
            end
        end

        % Construct the global left singular vectors
        U_k = []; % Initialize global left singular vector matrix
        for i = 1:num_clients
            U_k = [U_k; U_k_clients{i}]; % Stack the left singular vectors from all clients
        end
    else
        % Without client-side correction, directly construct the global left singular vectors
        U_k = []; % Initialize global left singular vector matrix
        for i = 1:num_clients
            U_k = [U_k; Q_Y_clients{i} * Q_R_clients{i} * U_tilde_k]; % Stack the uncorrected left singular vectors
        end
    end

    % Privacy Protection Analysis
    % Call the analyze_privacy_protection function to analyze the effect of ill-conditioned projection and randomization on privacy
    privacy_info = analyze_privacy_protection(A_clients, Omega_ill, R_Y_clients,B, params);
end