function [sol, info] = proxsgdtaublind(U, V, b, gamma, beta, init_z, maxiter, tol, ...
    early_stop, alpha_0, taups, show_info)

% Get problem size
[m, n] = size(U);

xidx = 1:n;
yidx = (n + 1) : (2 * n);

% Get initial point
z_before = init_z;
z_after = z_before;
bestz = init_z;

objs = zeros(m * maxiter + 1, 1);
bestobjs = zeros(m * maxiter + 1, 1);
obj = getblObj(U, V, b, z_before(xidx), z_before(yidx));
bestobj = obj;
objs = objs + obj;
bestobjs= bestobjs + bestobj;

% Number of epochs before reaching tolerance
nepochs = maxiter;
nbatchiter = maxiter * m;

% Initialize info struct
info.status = "x";

% Preprocess delays
ndelays = length(taups);
% assert(sum(taups) == 1.0);
cumps = cumsum(taups);
delayedx = cell(ndelays, 1);

for i = 1:ndelays
    delayedx{i} = init_z;
end % End for

if show_info
    fprintf("%6s %10s %10s %10s\n", 'epoch', 'obj', 'bobj', 'status');
end % End if

idx = 0; 
niter = m;
for k = 1:maxiter
    
    if bestobj < tol && nepochs == maxiter
        nepochs = k;
        niter = k * m + idx;
        info.status = "*";
        if early_stop
            if show_info
                fprintf("%6d %10.2e %10.2e %10s\n", k, obj, bestobj, info.status);
                fprintf("Early stopped. Status: Optimal\n");
                bestobjs((k - 1) * niter + 1:end) = bestobj;
                objs((k - 1) * niter + 1:end) = obj;
            end % End if
            break;
        end
    end % End if
    
    idx = 0;
    niter = m;
    
    for i = randperm(niter)
        
        idx = idx + 1;
        
        % Sample from dataset
        u = U(i, :);
        v = V(i, :);
        
        % Update momentum
        w = (1 + beta) * z_after - beta * z_before;
                
        % Get delay
        tau = gendelay(cumps);
        ztarget = delayedx{tau};
        z_before = z_after;
        
        uTx = u * ztarget(xidx);
        vTy = v * ztarget(yidx);
        sgn = sign(uTx .* vTy - b(i));
        sgn = sgn + (2 * rand() - 1) * (1 - abs(sgn));
        subgrad = [vTy * u'; uTx * v'] * sgn;
        delayedx = updatedelay(delayedx, z_after);
        z_after = w - (alpha_0 * subgrad / gamma);
        obj = getblObj(U, V, b, z_after(xidx), z_after(yidx));
        
        if obj < bestobj
            bestobj = obj;
            bestz = z_after;
        end % End if
        
        bestobjs((k * m - m) + idx + 1) = bestobj;
        objs((k * m - m) + idx + 1) = obj;
        
        if isnan(obj) || isinf(obj)
            info.status = "Diverged";
            break;
        end % End if
        
    end % End for
    
    if isnan(obj) || isinf(obj)
        break;
    end % End if
    
   if show_info && (mod(k, 50) == 0 || k == 1)
        fprintf("%6d %10.2e %10.2e %10s\n", k, obj, bestobj, info.status);
    end % End if
    
end % End for

% Collect information

% Solution array
sol.x = z_after;
sol.bestz = bestz;

% Information array
info.nepochs = nepochs;
info.niter = niter;
info.objs = objs;
info.bestobjs = bestobjs;

% Display summary
if show_info
    if info.status == "*"
        disp("- Algorithm reaches optimal after " + nepochs + " epochs (" + ...
            niter + " iterations)");
    elseif info.status == "x"
        disp("- Algorithm fails to reach desired accuracy after " +...
            nepochs + " epochs");
        info.niter = maxiter * m;
    elseif info.status == "Diverged"
        disp("- Algorithm diverges");
        info.niter = maxiter * m;
    end % End if 
end % End if

end % End function