using Krylov
using LinearAlgebra, Printf

m = 5
n = 8
λ = 1.0e-3
A = rand(m, n)
b = A * ones(n)
xy_exact = [A  λ*I] \ b # In Julia, this is the min-norm solution!

(x, y, stats) = craig(A, b, λ=λ, atol=0.0, rtol=1.0e-20, verbose=1)
show(stats)

# Check that we have a minimum-norm solution.
# When λ > 0 we solve min ‖(x,s)‖  s.t. Ax + λs = b, and we get s = λy.
@printf("Primal feasibility: %7.1e\n", norm(b - A * x - λ^2 * y) / norm(b))
@printf("Dual   feasibility: %7.1e\n", norm(x - A' * y) / norm(x))
@printf("Error in x: %7.1e\n", norm(x - xy_exact[1:n]) / norm(xy_exact[1:n]))
if λ > 0.0
  @printf("Error in y: %7.1e\n", norm(λ * y - xy_exact[n+1:n+m]) / norm(xy_exact[n+1:n+m]))
end
CRAIG: system of 5 equations in 8 variables
    k       ‖r‖       ‖x‖       ‖A‖      κ(A)         α        β  timer
    0  9.84e+00  0.00e+00  0.00e+00  0.00e+00   ✗ ✗ ✗ ✗  ✗ ✗ ✗ ✗  0.00s
    1  3.51e-01  2.79e+00  3.52e+00  3.52e+00   3.5e+00  1.3e-01  0.00s
    2  1.30e-01  2.81e+00  3.72e+00  5.26e+00   1.1e+00  4.1e-01  0.00s
    3  1.48e-03  2.82e+00  3.77e+00  6.61e+00   6.6e-01  7.5e-03  0.00s
    4  2.02e-03  2.82e+00  3.85e+00  7.77e+00   4.5e-01  6.2e-01  0.00s
    5  5.07e-10  2.82e+00  3.92e+00  9.40e+00   7.1e-01  1.8e-07  0.00s

SimpleStats
 niter: 5
 solved: true
 inconsistent: false
 indefinite: false
 npcCount: 0
 residuals: []
 Aresiduals: []
 κ₂(A): []
 timer: 63.21μs
 status: solution good enough for the tolerances given
Primal feasibility: 5.2e-11
Dual   feasibility: 1.0e-16
Error in x: 5.1e-11
Error in y: 4.3e-11