function [w, loss, criterion,dist] = Prox_ADMM_Decentralized(Xdata, Ydata, lambda, beta1, beta2, option,P,opt)

rng('default');
numWorkers=length(Xdata); 
y_agent=cell(numWorkers,1);
x_agent=cell(numWorkers,1);
l_agent=cell(numWorkers,1);

n=0;
[~,d]=size(Xdata{1});
tempx=cell(numWorkers,1);
temps=cell(numWorkers,1);
for k=1:numWorkers
    y_agent{k}=zeros(d,1);
    x_agent{k}=zeros(d,1);
    l_agent{k}=zeros(d,1);
    temps{k}=zeros(d,1);
end
for k=1:numWorkers
    [temp,~]=size(Xdata{k});
    n=n+temp;
end


epoch=100; 
eval_every=1;

% User's option
%------------------------------------------------
if nargin > 4  %  then paramstruct is an argument
    if isfield(option,'epoch')
        epoch = option.epoch; 
    end
    if isfield(option,'penalty')
        penalty = option.penalty; 
    end 
    if isfield(option,'loss')
        obj = option.loss; 
    end
end
% Aggregate full dataset (only for loss/criterion evaluation)
X = vertcat(Xdata{:});
Y = vertcat(Ydata{:});

loss=zeros(epoch, 1);
criterion=loss;

dist=loss;
% l1-norm proximal mapping operator
if matches(penalty,'l1')
    % l1-norm proximal mapping operator
    Proxg=@(z, lambda) sign(z).*max(abs(z)-lambda,0);
else
    % l2-norm proximal mapping operator
    Proxg=@(z, lambda) z/(lambda+1);
end



% display iteration information
fprintf('----------------Training with {%3.0d} workers------------\n', numWorkers);
fprintf('   round       primal loss     relative KKT residual   \n');



for i=1:epoch    
    % On All Agents:
    for k=1:numWorkers
        tempx{k}=zeros(d,1);
        for j=1:numWorkers
            tempx{k}=tempx{k}+P(j,k)*x_agent{j};
        end
        temps{k}=temps{k}+beta2*(x_agent{k}-tempx{k});
    end
    for k=1:numWorkers
        bk=beta1*y_agent{k}+l_agent{k}+beta2*(x_agent{k}+tempx{k});
        ck=bk-temps{k};
        if matches(obj,'ls')
            x_agent{k}=(Xdata{k}'*Xdata{k}/n+(beta1+2*beta2)*eye(d))\(Xdata{k}'*Ydata{k}/n+ck);
        else
            x_agent{k}=prox_weighted_hinge(Xdata{k}, Ydata{k}, ck/(beta1+2*beta2), 1/(beta1+2*beta2),1/n);
        end

    end
       
    for k=1:numWorkers
        y_agent{k}=Proxg(x_agent{k}-1/beta1*l_agent{k}, lambda/(numWorkers*beta1));
    end
    for k=1:numWorkers
        l_agent{k}=l_agent{k}-beta1*(x_agent{k}-y_agent{k});
    end
    w=zeros(d,1);
    for k=1:numWorkers
        w=w+x_agent{k}/numWorkers;
    end
    v=zeros(d,1);
    for k=1:numWorkers
        v=v+y_agent{k}/numWorkers;
    end
    if matches(obj,'ls')

        if matches(penalty,'l1')
            primal = norm(Y-(X*w))^2/(2*n)+lambda*norm(w, 1);
            kkt = norm(w-Proxg(w-X'*(X*w-Y)/n,lambda ));
        else
            primal = norm(Y-(X*w))^2/(2*n)+lambda/2*norm(w)^2;
            kkt = norm(w-Proxg(w-X'*(X*w-Y)/n,lambda ));
        end

    else

        if matches(penalty,'l1')
            primal=mean(max(0, 1-Y.*(X*w)) )+lambda*norm(w, 1);
            kkt=primal;
        else
            primal=mean(max(0, 1-Y.*(X*w)) )+lambda/2*norm(w)^2;
            kkt=primal;
        end

    end
    loss(i)=primal;
    criterion(i)=kkt;
    for k=1:numWorkers
        dist(i)=dist(i)+norm(opt-x_agent{k});
    end
    if mod(i,eval_every)==0        
        fprintf('%5d\t   %5d\t   %5d\t %5d\t \n', ...
                i, primal, kkt,dist(i));
    end    
    
end

end


function u_opt = prox_weighted_hinge(X, y, v, tau, k)

    [n, d] = size(X);
    gamma = tau;

    H = zeros(d + n);
    H(1:d, 1:d) = (1/gamma) * eye(d);

    f = zeros(d + n, 1);
    f(1:d) = -(1/gamma) * v;
    f(d+1:end) = k * ones(n, 1);  

    A = zeros(2*n, d+n);
    b = zeros(2*n, 1);

    for i = 1:n
        A(i, 1:d) = -y(i) * X(i, :);
        A(i, d+i) = -1;
        b(i) = -1;
    end

    A(n+1:end, d+1:end) = -eye(n);
    b(n+1:end) = 0;

    opts = optimoptions('quadprog', ...
    'OptimalityTolerance', 1e-10, ...
    'ConstraintTolerance', 1e-10, ...
    'Display', 'None');
    z = quadprog(H, f, A, b, [], [], [], [], [], opts);

    u_opt = z(1:d);
end

