function [w, loss, criterion,dist] = Decentralized_ACC_ADMM(Xdata, Ydata, lambda, beta, option,P,opt)
rng('default');
numWorkers=length(Xdata); 
v_agent=cell(numWorkers,1);
u_agent=cell(numWorkers,1);
w_agent_1=cell(numWorkers,1);
w_agent_2=cell(numWorkers,1);
A=P*P;
n=0;
[~,d]=size(Xdata{1});

for k=1:numWorkers
    v_agent{k}=zeros(d,1);
    u_agent{k}=zeros(d,1);
    w_agent_1{k}=zeros(d,1);
    w_agent_2{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 
    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');

tempv=cell(numWorkers,1);
tempu=cell(numWorkers,1);
tempw=cell(numWorkers,1);

tempv1=cell(numWorkers,1);
tempu1=cell(numWorkers,1);
tempw1=cell(numWorkers,1);
tempw2=cell(numWorkers,1);
for i=1:epoch    
    for k=1:numWorkers
        tempv{k}=zeros(d,1);
        tempu{k}=zeros(d,1);
        tempw{k}=zeros(d,1);
        tempv1{k}=v_agent{k};
        tempu1{k}=u_agent{k};
        tempw1{k}=w_agent_1{k};
        tempw2{k}=w_agent_2{k};
        for j=1:numWorkers
            tempv{k}=tempv{k}+P(j,k)*v_agent{j};
            tempw{k}=tempw{k}+P(j,k)*w_agent_1{j};
            tempu{k}=tempu{k}+A(j,k)*u_agent{j};
        end
    end
    for k=1:numWorkers
        if matches(obj,'ls')
            u_agent{k}=(Xdata{k}'*Xdata{k}/n+2.01*beta*eye(d))\(Xdata{k}'*Ydata{k}/n+beta*(2*tempv{k}+1.01*u_agent{k}-tempu{k})-w_agent_2{k}+tempw{k});
        else
            u_agent{k}=prox_weighted_hinge(Xdata{k}, Ydata{k}, (2*tempv{k}+1.01*u_agent{k}-tempu{k})/2.01-w_agent_2{k}/(2.01*beta)+tempw{k}/(2.01*beta), 1/(2.01*beta),1/n);
        end
   
    end
    for k=1:numWorkers
        temp1=zeros(d,1);
        temp2=zeros(d,1);
        for j=1:numWorkers
            temp1=temp1+P(j,k)*u_agent{j};
            temp2=temp2+P(j,k)*v_agent{j};
        end
        w_agent_1{k}=w_agent_1{k}-0.99*beta*(temp1-v_agent{k});
        w_agent_2{k}=w_agent_2{k}-0.99*beta*(temp2-u_agent{k});
    end
 
    for k=1:numWorkers
        temp1=zeros(d,1);
        temp2=zeros(d,1);
        temp3=zeros(d,1);
        for j=1:numWorkers
            temp1=temp1+P(j,k)*w_agent_2{j};
            temp2=temp2+P(j,k)*u_agent{j};
            temp3=temp3+A(j,k)*v_agent{j};
        end
        v_agent{k}=Proxg((2*temp2-temp3+v_agent{k}+temp1/beta-w_agent_1{k}/beta)/2, lambda/(2*numWorkers*beta));
    end
    for k=1:numWorkers
        temp1=zeros(d,1);
        temp2=zeros(d,1);
        for j=1:numWorkers
            temp1=temp1+P(j,k)*u_agent{j};
            temp2=temp2+P(j,k)*v_agent{j};
        end
        w_agent_1{k}=w_agent_1{k}-beta*(temp1-v_agent{k});
        w_agent_2{k}=w_agent_2{k}-beta*(temp2-u_agent{k});
    end
    w=zeros(d,1);
    for k=1:numWorkers
        w=w+u_agent{k}/numWorkers;
    end
    v=zeros(d,1);
    for k=1:numWorkers
        v=v+v_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-u_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-12, ...
    'ConstraintTolerance', 1e-12, ...
    'Display', 'None');
    z = quadprog(H, f, A, b, [], [], [], [], [], opts);

    u_opt = z(1:d);
end

