function [w, loss, criterion,dist] = ProxMudag(Xdata, Ydata, lambda, eta, alpha, option,P,opt)

rng('default');
numWorkers=length(Xdata); 
v_agent=cell(numWorkers,1);
u_agent=cell(numWorkers,1);
w_agent=cell(numWorkers,1);

eta1=0;
n=0;
[~,d]=size(Xdata{1});

for k=1:numWorkers
    v_agent{k}=zeros(d,1);
    u_agent{k}=zeros(d,1);
end
for k=1:numWorkers
    [temp,~]=size(Xdata{k});
    n=n+temp;
end
for k=1:numWorkers
    w_agent{k}=(1/n)*Xdata{k}'*(Xdata{k}*v_agent{k}-Ydata{k});
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');

tempv=cell(numWorkers,1);
temps=cell(numWorkers,1);
tempw=cell(numWorkers,1);

for i=1:epoch    
    % On All Agents:
    tempv=v_agent;
    tempu=u_agent;
    for k=1:numWorkers
        if matches(obj,'ls')
            v_agent{k}=Proxg(u_agent{k}-eta*w_agent{k}, eta*lambda/numWorkers);
        else
            v_agent{k}=prox_weighted_hinge(Xdata{k}, Ydata{k}, u_agent{k}-eta*w_agent{k}, eta, lambda);
        end
   
    end
    temp1=cell(numWorkers,1);
    temp2=cell(numWorkers,1);
    temp3=cell(numWorkers,1);
    for k=1:numWorkers
        temp1{k}=v_agent{k}+(1-alpha)/(1+alpha)*(v_agent{k}-tempv{k});
        temp2{k}=zeros(d,1);
    end
    for k=1:numWorkers
        for j=1:numWorkers
            temp2{k}=temp2{k}+P(j,k)*temp1{j};
        end
        temp3{k}=zeros(d,1);
    end
    for k=1:numWorkers
        for j=1:numWorkers
            temp3{k}=temp3{k}+(1+eta1)*(P(j,k)*temp2{j});
        end
        temp3{k}=temp3{k}-eta1*temp1{k};
    end
    u_agent=temp3;
    if matches(obj,'ls')
    for k=1:numWorkers
        temp1{k}=w_agent{k}+(1/n)*Xdata{k}'*(Xdata{k}*u_agent{k}-Ydata{k})-(1/n)*Xdata{k}'*(Xdata{k}*tempu{k}-Ydata{k});
        temp2{k}=zeros(d,1);
    end
    else
        for k=1:numWorkers
        temp1{k}=w_agent{k}+(lambda/numWorkers)*(u_agent{k}-tempu{k});
        temp2{k}=zeros(d,1);
        end
    end
    for k=1:numWorkers
        for j=1:numWorkers
            temp2{k}=temp2{k}+P(j,k)*temp1{j};
        end
        temp3{k}=zeros(d,1);
    end
    for k=1:numWorkers
        for j=1:numWorkers
            temp3{k}=temp3{k}+(1+eta1)*(P(j,k)*temp2{j});
        end
        temp3{k}=temp3{k}-eta1*temp1{k};
    end
    w_agent=temp3;
    
    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-10, ...
    'ConstraintTolerance', 1e-10, ...
    'Display', 'None');
    z = quadprog(H, f, A, b, [], [], [], [], [], opts);
    
    u_opt = z(1:d);
end