using CombDiff, Test

@pct (x::N) -> indicator(x, 4, x)

f, _ = @pct (i::N, j::N) -> vac_exp(:a(i) ∘ :a(i)')
f, _ = @pct (g::CM) -> sum((i, j), g(i, j) * vac_exp(:a(i) ∘ :a(j)'))

simplify(reduce_vac(f)) |> first

f, _ = @pct (N_o::N, g::CM) -> ((a::FField) -> sum((i, j), vac_exp(:II(g(i, j)) ∘ a(i) ∘ a(j)')))(
    (p::N) -> 𝕀(p, N_o, :b(p)') + 𝕀(N_o + 1, p, :c(p))
)

@macroexpand @pct (N_o::N, g::CM) -> ((a::FField) -> sum((i, j), vac_exp(:II(g(i, j)) ∘ a(i) ∘ a(j)')))(
    (p::N) -> 𝕀(p, N_o, :b(p)') + 𝕀(N_o + 1, p, :c(p))
)

f, _ = @pct (g::CM, i::N, j::N) -> g(i, j) * vac_exp(:a(i) ∘ :a(j)')

f, _ = @pct (i::N, j::N) -> vac_exp((:a(i) + :a(j)') ∘ (:a(j) + :a(i)'))

f, _ = @pct (i::N, j::N) -> vac_exp(i * :a(j))


f, _ = @pct (i::N, j::N) -> ((a::FField) -> a(i) ∘ a(j)')(:b)

# f, _ = @pct (N_o::N) -> ((c::FField) -> 𝕀 )
"""

"""


simplify(reduce_vac(f)) |> first

vacuum_exp(content(get_body(get_body(f))))
x, y = content(get_body(get_body(f)))
anti_commutator(x, y)

vacuum_exp([x, y])

f, _ = @pct (i::N, j::N) -> :a(i) ∘ :a(j)'
vacuum_exp(content(get_body(get_body(f))))

f, _ = @pct (i::N, j::N) -> :a(i) ∘ :b(j)'
vacuum_exp(content(get_body(get_body(f))))


f, _ = @pct (ϵ::RV) -> sum((i::N ∈ (1, N)), ϵ(i) * (:a(i) ∘ :a(i)'))
vacuum_exp(f)

using CombDiff

fx, ctx = @pct begin
    @space T begin
        type = (I, I, I, I) -> C
        symmetries = (((2, 1, 4, 3), :conj), ((3, 4, 1, 2), :id))
    end
    (J::T) -> _
end

e2, _ = @pct fx ctx (N_o::N, v::T) -> ((a::FField) -> sum((i, j, k, l), vac_exp(:II(v(i, j, k, l)) ∘ a(i)' ∘ a(j)' ∘ a(l) ∘ a(k))))(
    (p::N) -> 𝕀(p, N_o, :b(p)') + 𝕀(N_o + 1, p, :c(p))
);
e2

@profview g = simplify(eval_all(e2); settings=custom_settings(:logging => false))

f, _ = @pct (N_o::N, g::CM) -> ((a::FField) ->
    sum((i, j), vac_exp(:II(g(i, j)) ∘ a(i) ∘ a(j)')))(
    (p::N) -> 𝕀(p, N_o, :b(p)') + 𝕀(N_o + 1, p, :c(p))
);
f

@profview g = f |> eval_all |> simplify |> first

e2, _ = @pct fx ctx begin
    H = ((a::FField) -> sum((i, j, k, l), :II(v(i, j, k, l)) ∘ a(i)' ∘ a(j)' ∘ a(l) ∘ a(k)))(
        (p::N) -> 𝕀(p, N_o, :d(p)') + 𝕀(N_o + 1, p, :c(p)))
    ψ = :II(1) + sum((i, a),
            𝕀(i, N_o, 𝕀(N_o + 1, a), :II(t1(i, a)) ∘ :d(i)' ∘ :c(a)')) +
        sum((i, j, a, b), 𝕀(i, N_o, 𝕀(j, N_o, 𝕀(N_o + 1, a, 𝕀(N_o + 1, b, :II(t2(i, j, a, b)) ∘ :d(i)' ∘ :d(j)' ∘ :c(a)' ∘ :c(b)')))))
    vac_exp(ψ' ∘ H ∘ ψ)
end

e2

using CombDiff
fx, ctx = @pct begin
    @space T begin
        type = (N, N, N, N) -> R
        symmetries = (((2, 1, 3, 4), :neg), ((1, 2, 4, 3), :neg))
    end
    @space V begin
        type = (N, N, N, N) -> R
        symmetries = (((2, 1, 4, 3), :id), ((3, 4, 1, 2), :id))
    end
    (N_o::N, h::Sym, v::V, t₁::RM, t₂::T) -> _

end;

e2, _ = @pct fx ctx begin
    H = ((a::FField) ->
        sum((r, s), :II(h(r, s)) ∘ a(r) ∘ a(s)'))((p::N) -> 𝕀(p, N_o, :d(p)') + 𝕀(N_o + 1, p, :c(p)))
    ψ = :II(1) + sum((i, a), 𝕀(i, N_o, 𝕀(N_o + 1, a, :II(t₁(i, a)) ∘ :d(i)' ∘ :c(a)')))
    #= + 
        sum((i, j, a, b), 𝕀(i, N_o, 𝕀(j, N_o, 𝕀(N_o + 1, a, 𝕀(N_o + 1, b, :II(t₂(i, j, a, b)) ∘ :d(i)' ∘ :d(j)' ∘ :c(a)' ∘ :c(b)'))))) =#
    vac_exp(ψ' ∘ H ∘ ψ)
end

e2, _ = @pct fx ctx begin
    H = ((a::FField) ->
        sum((r, s), :II(h(r, s)) ∘ a(r)' ∘ a(s)))((p::N) -> 𝕀(p, N_o, :d(p)') + 𝕀(N_o + 1, p, :c(p)))
    ψ = :II(1) + sum((i, a), 𝕀(i, N_o, 𝕀(N_o + 1, a, :II(t₁(i, a)) ∘ :d(i)' ∘ :c(a)'))) +
        sum((i, j, a, b), 𝕀(i, N_o, 𝕀(j, N_o, 𝕀(N_o + 1, a, 𝕀(N_o + 1, b, :II(t₂(i, j, a, b)) ∘ :d(i)' ∘ :d(j)' ∘ :c(a)' ∘ :c(b)')))))
    vac_exp(ψ' ∘ ψ)
end

e2e = eval_all(e2)

l = Logger()
g = simplify(e2e; logger=l) |> first
res = symmetry_reduction(g; logger=l)
mathjax(res)
symmetry_reduction(res)
g_s = simplify(g; logger=l, settings=custom_settings(:expand_mul => true, :gcd => false; preset=symmetry_settings()))
html_report(l)

f, _ = @pct (i::N, j::N, a::N, b::N, ip::N, jp::N, ap::N, bp::N) -> vac_exp(:c(b) ∘ :c(a) ∘ :d(j) ∘ :d(i) ∘ :d(ip)' ∘ :d(jp)' ∘ :c(ap)' ∘ :c(bp)')

simplify(f; settings=custom_settings(:expand_mul => true, :gcd => false)) |> first |> mathjax

f, _ = @pct fx ctx (i::N, j::N, a::N, b::N) -> t₂(i, j, a, b)

# CCD Energy

e3, _ = @pct fx ctx begin
    let H = ((a::FField) ->
            sum((r, s), :II(h(r, s)) ∘ a(r)' ∘ a(s)) + sum((i, j, k, l), :II(v(i, j, k, l)) ∘ a(i)' ∘ a(j)' ∘ a(l) ∘ a(k))
        )((p::N) -> 𝕀(p, N_o, :d(p)') + 𝕀(N_o + 1, p, :c(p)))
        let T2 = :II(0.25) ∘ sum((i, j, a, b), 𝕀(i, N_o, 𝕀(j, N_o, 𝕀(N_o + 1, a, 𝕀(N_o + 1, b, :II(t₂(i, j, a, b)) ∘ :d(i)' ∘ :d(j)' ∘ :c(a)' ∘ :c(b)')))))
            let ψ = :II(1) + T2 + :II(0.5) ∘ T2 ∘ T2
                vac_exp(H ∘ ψ)
            end
        end
    end
end

e3e = eval_all(e3)
g = simplify(e3e) |> first
res = symmetry_reduction(g)

# CCD Equations (double)
fx, ctx = @pct begin
    @space T begin
        type = (N, N, N, N) -> R
        symmetries = (((2, 1, 3, 4), :neg), ((1, 2, 4, 3), :neg))
    end
    @space V begin
        type = (N, N, N, N) -> R
        symmetries = (((2, 1, 3, 4), :neg), ((1, 2, 4, 3), :neg), ((3, 4, 1, 2), :id))
    end
    (N_o::N, h::Sym, v::V, t₁::RM, t₂::T) -> _

end;

e3, _ = @pct fx ctx begin
    (a::N ∈ [N_o + 1, ∞], b::N ∈ [N_o + 1, ∞], i::N ∈ [1, N_o], j::N ∈ [1, N_o]) ->
        let H = ((a::FField) ->
            #= sum((r, s), :II(h(r, s)) ∘ a(r)' ∘ a(s))  =#
                sum((r, s, k, l), :II(v(r, s, k, l)) ∘ a(r)' ∘ a(s)' ∘ a(k) ∘ a(l))
            )((p::N) -> 𝕀(p, N_o, :d(p)') + 𝕀(N_o + 1, p, :c(p)))
            let T2 = :II(0.25) ∘ sum((i, j, a, b), 𝕀(i, N_o, 𝕀(j, N_o, 𝕀(N_o + 1, a, 𝕀(N_o + 1, b, :II(t₂(i, j, a, b)) ∘ :d(i)' ∘ :d(j)' ∘ :c(a)' ∘ :c(b)')))))
                let ψ = :II(1) + T2 + :II(0.5) ∘ T2 ∘ T2
                    vac_exp(:c(a) ∘ :c(b) ∘ :d(i) ∘ :d(j) ∘ H ∘ ψ)
                end
            end
        end
end

l = Logger()
e3e = eval_all(e3)
g = simplify(e3e) |> first
res = simplify(fast_symmetry_reduction(g); settings=custom_settings(:expand_mul => true, :gcd => false)) |> first


e3, _ = @pct fx ctx begin
    (a::N ∈ [N_o + 1, ∞], b::N ∈ [N_o + 1, ∞], i::N ∈ [1, N_o], j::N ∈ [1, N_o]) ->
        ((a::FField) ->
        #= sum((r, s), :II(h(r, s)) ∘ a(r)' ∘ a(s))  =#
            sum((r, s, k, l), :II(v(r, s, k, l)) ∘ a(r)' ∘ a(s)' ∘ a(k) ∘ a(l))
        )((p::N) -> 𝕀(p, N_o, :d(p)') + 𝕀(N_o + 1, p, :c(p)))
end

e3e = eval_all(e3)

g = simplify(e3e; settings=custom_settings(:expand_comp => true, :dist_conj => true)) |> first

