Skip to content
This repository has been archived by the owner on Jul 7, 2024. It is now read-only.

Extend Tenet.contract to contract :between two Sites #30

Merged
merged 15 commits into from
Mar 14, 2024
Merged
23 changes: 23 additions & 0 deletions src/Ansatz/Chain.jl
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
using Tenet
using LinearAlgebra
using Random
using ValSplit
using Muscle: gramschmidt!

struct Chain <: Ansatz
Expand Down Expand Up @@ -227,6 +228,28 @@ function Base.rand(rng::Random.AbstractRNG, sampler::ChainSampler, ::Type{Open},
Chain(Operator(), Open(), arrays)
end

@valsplit 2 Tenet.contract(tn::Chain, query::Symbol, site1::Site, site2::Site; direction::Symbol = :left) =
Tenet.contract!(copy(tn), Val(query), site1, site2; direction=direction)

function Tenet.contract!(tn::Chain, ::Val{:between}, site1::Site, site2::Site; direction::Symbol = :left)
jofrevalles marked this conversation as resolved.
Show resolved Hide resolved
Λᵢ = select(tn, :between, site1, site2)
Λᵢ === nothing && return tn

Λᵢ = pop!(TensorNetwork(tn), Λᵢ)

if direction === :right
Γᵢ₊₁ = select(tn, :tensor, site2)
replace!(TensorNetwork(tn), Γᵢ₊₁ => contract(Γᵢ₊₁, Λᵢ, dims = ()))
elseif direction === :left
Γᵢ = select(tn, :tensor, site1)
replace!(TensorNetwork(tn), Γᵢ => contract(Λᵢ, Γᵢ, dims = ()))
else
throw(ArgumentError("Unknown direction=:$direction"))
end

jofrevalles marked this conversation as resolved.
Show resolved Hide resolved
return tn
end

canonize_site(tn::Chain, args...; kwargs...) = canonize_site!(deepcopy(tn), args...; kwargs...)
canonize_site!(tn::Chain, args...; kwargs...) = canonize_site!(boundary(tn), tn, args...; kwargs...)

Expand Down
49 changes: 40 additions & 9 deletions test/Ansatz/Chain_test.jl
Original file line number Diff line number Diff line change
Expand Up @@ -107,6 +107,31 @@
@testset "Canonization" begin
using Tenet

@testset "contract" begin
qtn = rand(Chain, Open, State; n = 5, p = 2, χ = 20)
canonized = canonize(qtn)

for i in 1:4
contract_some = contract(canonized, :between, Site(i), Site(i + 1))
Bᵢ = select(contract_some, :tensor, Site(i))

@test isapprox(contract(TensorNetwork(contract_some)), contract(TensorNetwork(qtn)))

@test length(tensors(canonized)) == length(tensors(contract_some)) + 1 # We removed one singular value vector
jofrevalles marked this conversation as resolved.
Show resolved Hide resolved
@test_throws ArgumentError select(contract_some, :between, Site(i), Site(i + 1))

@test isrightcanonical(contract_some, Site(i))
@test isleftcanonical(
contract(canonized, :between, Site(i), Site(i + 1); direction = :right),
Site(i + 1),
)

Γᵢ = select(canonized, :tensor, Site(i))
Λᵢ₊₁ = select(canonized, :between, Site(i), Site(i + 1))
@test Bᵢ ≈ contract(Γᵢ, Λᵢ₊₁; dims = ())
end
end

@testset "canonize_site" begin
qtn = Chain(State(), Open(), [rand(4, 4), rand(4, 4, 4), rand(4, 4)])

Expand Down Expand Up @@ -164,28 +189,34 @@
Λ = [select(canonized, :between, Site(i), Site(i + 1)) for i in 1:4]
@test map(λ -> sum(abs2, λ), Λ) ≈ ones(length(Λ)) * norm(canonized)^2

for i in 1:4
for i in 1:5
canonized = canonize(qtn)

if i == 1
@test isleftcanonical(canonized, Site(i))
elseif i == 5 # in the limits of the chain, we get the norm of the state
contract!(canonized, :between, Site(i - 1), Site(i); direction = :right)
tensor = select(canonized, :tensor, Site(i))
replace!(TensorNetwork(canonized), tensor => tensor / norm(canonized))
@test isleftcanonical(canonized, Site(i))
else
Γᵢ = select(canonized, :tensor, Site(i))
Λᵢ = pop!(TensorNetwork(canonized), select(canonized, :between, Site(i - 1), Site(i)))
replace!(TensorNetwork(canonized), Γᵢ => contract(Λᵢ, Γᵢ; dims = ()))
contract!(canonized, :between, Site(i - 1), Site(i); direction = :right)
@test isleftcanonical(canonized, Site(i))
end
end

for i in 2:5
for i in 1:5
canonized = canonize(qtn)

if i == 5
if i == 1 # in the limits of the chain, we get the norm of the state
contract!(canonized, :between, Site(i), Site(i + 1); direction = :left)
tensor = select(canonized, :tensor, Site(i))
replace!(TensorNetwork(canonized), tensor => tensor / norm(canonized))
@test isrightcanonical(canonized, Site(i))
elseif i == 5
@test isrightcanonical(canonized, Site(i))
else
Γᵢ = select(canonized, :tensor, Site(i))
Λᵢ₊₁ = pop!(TensorNetwork(canonized), select(canonized, :between, Site(i), Site(i + 1)))
replace!(TensorNetwork(canonized), Γᵢ => contract(Γᵢ, Λᵢ₊₁; dims = ()))
contract!(canonized, :between, Site(i), Site(i + 1); direction = :left)
@test isrightcanonical(canonized, Site(i))
end
end
Expand Down
Loading