From f99e2e4d2e5189945cab841415dc7e6d319b4acf Mon Sep 17 00:00:00 2001 From: Ronny Bergmann Date: Wed, 23 Oct 2024 20:46:02 +0200 Subject: [PATCH 1/6] introdice invertible matrices (merely for completeness) --- NEWS.md | 8 +- Project.toml | 2 +- docs/make.jl | 1 + docs/src/index.md | 2 +- docs/src/manifolds/invertible.md | 7 ++ .../config/vocabularies/Manifolds/accept.txt | 3 + src/Manifolds.jl | 6 +- src/manifolds/InvertibleMatrices.jl | 116 ++++++++++++++++++ test/manifolds/invertible_matrices.jl | 36 ++++++ test/runtests.jl | 1 + 10 files changed, 176 insertions(+), 6 deletions(-) create mode 100644 docs/src/manifolds/invertible.md create mode 100644 src/manifolds/InvertibleMatrices.jl create mode 100644 test/manifolds/invertible_matrices.jl diff --git a/NEWS.md b/NEWS.md index b405b8f99c..741b7be6b8 100644 --- a/NEWS.md +++ b/NEWS.md @@ -5,11 +5,15 @@ All notable changes to this project will be documented in this file. The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). -## [0.10.5] – unreleased +## [0.10.5] - 2024-10-24 + +### Added + +* the manifold `InvertibleMatrices` of invertible matrices ### Changed -* rewrote the `CONTRIBUTING.md` and adapt it to todays links and references. +* rewrote the `CONTRIBUTING.md` and adapt it to today's links and references. ## [0.10.4] - 2024-10-20 diff --git a/Project.toml b/Project.toml index 7aa1553644..4c7f1e010d 100644 --- a/Project.toml +++ b/Project.toml @@ -1,7 +1,7 @@ name = "Manifolds" uuid = "1cead3c2-87b3-11e9-0ccd-23c62b72b94e" authors = ["Seth Axen ", "Mateusz Baran ", "Ronny Bergmann ", "Antoine Levitt "] -version = "0.10.4" +version = "0.10.5" [deps] Einsum = "b7d42ee7-0b51-5a75-98ca-779d3107e4c0" diff --git a/docs/make.jl b/docs/make.jl index 8807884d80..6a5b69232f 100755 --- a/docs/make.jl +++ b/docs/make.jl @@ -129,6 +129,7 @@ makedocs(; "Hamiltonian" => "manifolds/hamiltonian.md", "Hyperbolic space" => "manifolds/hyperbolic.md", "Hyperrectangle" => "manifolds/hyperrectangle.md", + "Invertible matrices" => "manifolds/invertible.md", "Lorentzian manifold" => "manifolds/lorentz.md", "Multinomial doubly stochastic matrices" => "manifolds/multinomialdoublystochastic.md", "Multinomial matrices" => "manifolds/multinomial.md", diff --git a/docs/src/index.md b/docs/src/index.md index 1a386ab191..59a44d220d 100644 --- a/docs/src/index.md +++ b/docs/src/index.md @@ -44,7 +44,7 @@ If you use `Manifolds.jl` in your work, please cite the following } ``` -To refer to a certain version we recommend to also cite for example +To refer to a specific version, it is recommended to cite, for example, ```biblatex @software{manifoldsjl-zenodo-mostrecent, diff --git a/docs/src/manifolds/invertible.md b/docs/src/manifolds/invertible.md new file mode 100644 index 0000000000..b7a1656a9e --- /dev/null +++ b/docs/src/manifolds/invertible.md @@ -0,0 +1,7 @@ +# Invertible matrices + +```@autodocs +Modules = [Manifolds] +Pages = ["manifolds/InvertibleMatrices.jl"] +Order = [:type, :function] +``` diff --git a/docs/styles/config/vocabularies/Manifolds/accept.txt b/docs/styles/config/vocabularies/Manifolds/accept.txt index a6ca4b1ddb..8596c7e5ba 100644 --- a/docs/styles/config/vocabularies/Manifolds/accept.txt +++ b/docs/styles/config/vocabularies/Manifolds/accept.txt @@ -1,2 +1,5 @@ +Grassmann Riemannian +Stiefel +[sS]ymplectic struct \ No newline at end of file diff --git a/src/Manifolds.jl b/src/Manifolds.jl index 589cd9c5c2..5c81913a59 100644 --- a/src/Manifolds.jl +++ b/src/Manifolds.jl @@ -397,7 +397,7 @@ include("manifolds/VectorFiber.jl") include("manifolds/VectorBundle.jl") include("groups/group.jl") -# Features I: Which are extended on Meta Manifolds +# Features I: Extending Meta Manifolds include("statistics.jl") # Meta Manifolds II: Products @@ -415,7 +415,7 @@ METAMANIFOLDS = [ VectorBundle, ] -# Features II: That require metas +# Features II: That require MetaManifolds include("atlases.jl") include("differentiation/ode_callback.jl") include("cotangent_space.jl") @@ -443,6 +443,7 @@ include("manifolds/GeneralizedGrassmann.jl") include("manifolds/GeneralizedStiefel.jl") include("manifolds/Hyperbolic.jl") include("manifolds/Hyperrectangle.jl") +include("manifolds/InvertibleMatrices.jl") include("manifolds/MultinomialDoublyStochastic.jl") include("manifolds/MultinomialSymmetric.jl") include("manifolds/MultinomialSymmetricPositiveDefinite.jl") @@ -681,6 +682,7 @@ export Euclidean, HeisenbergGroup, Hyperbolic, Hyperrectangle, + InvertibleMatrices, KendallsPreShapeSpace, KendallsShapeSpace, Lorentz, diff --git a/src/manifolds/InvertibleMatrices.jl b/src/manifolds/InvertibleMatrices.jl new file mode 100644 index 0000000000..f99a800446 --- /dev/null +++ b/src/manifolds/InvertibleMatrices.jl @@ -0,0 +1,116 @@ +@doc raw""" + InvertibleMatrices{𝔽,T} <: AbstractDecoratorManifold{𝔽} + +The [`AbstractManifold`](@extref `ManifoldsBase.AbstractManifold`) ``\operatorname{Sym}(n)`` +consisting of the real- or complex-valued invertible matrices, that is the set + +```math +\bigl\{p ∈ 𝔽^{n×n}\ \big|\ \det(p) \neq 0 \bigr\}, +``` +where the field ``𝔽 ∈ \{ ℝ, ℂ\}``. + +# Constructor + + InvertibleMatrices(n::Int, field::AbstractNumbers=ℝ) + +Generate the manifold of ``n×n`` invertible matrices. +""" +struct InvertibleMatrices{𝔽,T} <: AbstractDecoratorManifold{𝔽} + size::T +end + +function InvertibleMatrices(n::Int, field::AbstractNumbers=ℝ; parameter::Symbol=:type) + size = wrap_type_parameter(parameter, (n,)) + return InvertibleMatrices{field,typeof(size)}(size) +end + +function active_traits(f, ::InvertibleMatrices, args...) + return merge_traits(IsEmbeddedSubmanifold()) +end + +function allocation_promotion_function( + ::InvertibleMatrices{ℂ,<:Any}, + ::typeof(get_vector), + args::Tuple, +) + return complex +end + +@doc raw""" + check_point(M::InvertibleMatrices{n,𝔽}, p; kwargs...) + +Check whether `p` is a valid manifold point on the [`InvertibleMatrices`](@ref) `M`, i.e. +whether `p` is a symmetric matrix of size `(n,n)` with values from the corresponding +[`AbstractNumbers`](@extref ManifoldsBase number-system) `𝔽`. + +The tolerance for the symmetry of `p` can be set using `kwargs...`. +""" +function check_point(M::InvertibleMatrices, p; kwargs...) + if det(p) == 0 + return DomainError( + det(p), + "The point $(p) does not lie on $(M), since its determinant is zero and hence it is not invertible.", + ) + end + return nothing +end + +""" + check_vector(M::InvertibleMatrices{n,𝔽}, p, X; kwargs... ) + +Check whether `X` is a tangent vector to manifold point `p` on the +[`InvertibleMatrices`](@ref) `M`, which are all matrces of size ``n×n``` +its values have to be from the correct [`AbstractNumbers`](@extref ManifoldsBase number-system). +""" +function check_vector(M::InvertibleMatrices, p, X; kwargs...) + return nothing +end + +embed(::InvertibleMatrices, p) = p +embed(::InvertibleMatrices, p, X) = X + +function get_embedding(::InvertibleMatrices{𝔽,TypeParameter{Tuple{n}}}) where {n,𝔽} + return Euclidean(n, n; field=𝔽) +end +function get_embedding(M::InvertibleMatrices{𝔽,Tuple{Int}}) where {𝔽} + n = get_parameter(M.size)[1] + return Euclidean(n, n; field=𝔽, parameter=:field) +end + +""" + is_flat(::InvertibleMatrices) + +Return true. [`InvertibleMatrices`](@ref) is a flat manifold. +""" +is_flat(M::InvertibleMatrices) = true + +@doc raw""" + manifold_dimension(M::InvertibleMatrices{n,𝔽}) + +Return the dimension of the [`InvertibleMatrices`](@ref) matrix `M` over the number system +`𝔽`, which is the same dimension as its embedding, the [`Euclidean`](@ref)`(n,n)`. +""" +function manifold_dimension(M::InvertibleMatrices{<:Any,𝔽}) where {𝔽} + return manifold_dimension(get_embedding(M)) +end + +function Base.show(io::IO, ::InvertibleMatrices{𝔽,TypeParameter{Tuple{n}}}) where {n,𝔽} + return print(io, "InvertibleMatrices($(n), $(𝔽))") +end +function Base.show(io::IO, M::InvertibleMatrices{𝔽,Tuple{Int}}) where {𝔽} + n = get_parameter(M.size)[1] + return print(io, "InvertibleMatrices($(n), $(𝔽); parameter=:field)") +end + +@doc raw""" + Y = Weingarten(M::InvertibleMatrices, p, X, V) + Weingarten!(M::InvertibleMatrices, Y, p, X, V) + +Compute the Weingarten map ``\mathcal W_p`` at `p` on the [`InvertibleMatrices`](@ref) `M` with respect to the +tangent vector ``X \in T_p\mathcal M`` and the normal vector ``V \in N_p\mathcal M``. + +Since this a flat space by itself, the result is always the zero tangent vector. +""" +Weingarten(::InvertibleMatrices, p, X, V) + +Weingarten!(::InvertibleMatrices, Y, p, X, V) = fill!(Y, 0) diff --git a/test/manifolds/invertible_matrices.jl b/test/manifolds/invertible_matrices.jl new file mode 100644 index 0000000000..27ab5a6d98 --- /dev/null +++ b/test/manifolds/invertible_matrices.jl @@ -0,0 +1,36 @@ +using LinearAlgebra, Manifolds, ManifoldsBase, Test + +@testset "Invertible matrices" begin + M = InvertibleMatrices(3, ℝ) + A = [1.0 0.0 0.0; 0.0 1.0 0.0; 0.0 0.0 1.0] + B = [0.0 0.0 0.0; 0.0 1.0 0.0; 0.0 0.0 1.0] + Mc = InvertibleMatrices(3, ℂ) + Ac = [1.0im 0.0 0.0; 0.0 1.0 0.0; 0.0 0.0 1.0] + Bc = [0.0im 0.0 0.0; 0.0 1.0 0.0; 0.0 0.0 1.0] + @testset "Real invertible matrices" begin + @test repr(M) == "InvertibleMatrices(3, ℝ)" + M2 = InvertibleMatrices(3, ℝ; parameter=:field) + @test repr(M2) == "InvertibleMatrices(3, ℝ; parameter=:field)" + @test check_point(M, A) == nothing + @test_throws DomainError is_point(M, B; error=:error) + @test_throws ManifoldDomainError is_point(M, Ac; error=:error) + @test_throws ManifoldDomainError is_vector(M, A, Ac; error=:error) + @test is_flat(M) + @test typeof(get_embedding(M)) === Euclidean{TypeParameter{Tuple{3,3}},ℝ} + @test typeof(get_embedding(M2)) === Euclidean{Tuple{Int64,Int64},ℝ} + @test embed(M, A) === A + @test embed(M, A, A) === A + @test manifold_dimension(M) == 9 + @test Weingarten(M, A, A, A) == zero(A) + end + @testset "Complex invertible matrices" begin + @test repr(Mc) == "InvertibleMatrices(3, ℂ)" + Mc2 = InvertibleMatrices(3, ℂ; parameter=:field) + @test repr(Mc2) == "InvertibleMatrices(3, ℂ; parameter=:field)" + @test manifold_dimension(Mc) == 2*3^2 + @test check_point(Mc, Ac) == nothing + @test_throws DomainError is_point(Mc, Bc; error=:error) + @test_throws DomainError is_point(Mc, B; error=:error) + @test is_point(Mc, A; error=:error) + end +end \ No newline at end of file diff --git a/test/runtests.jl b/test/runtests.jl index f6bd62c799..49b82cea64 100644 --- a/test/runtests.jl +++ b/test/runtests.jl @@ -152,6 +152,7 @@ end include_test("manifolds/hamiltonian.jl") include_test("manifolds/hyperbolic.jl") include_test("manifolds/hyperrectangle.jl") + include_test("manifolds/invertible_matrices.jl") include_test("manifolds/lorentz.jl") include_test("manifolds/multinomial_doubly_stochastic.jl") include_test("manifolds/multinomial_symmetric.jl") From 6eda88487d03fe9979a85e04247f9e0a3a0c2c0d Mon Sep 17 00:00:00 2001 From: Ronny Bergmann Date: Wed, 23 Oct 2024 20:56:02 +0200 Subject: [PATCH 2/6] runs formatter. --- test/manifolds/invertible_matrices.jl | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/test/manifolds/invertible_matrices.jl b/test/manifolds/invertible_matrices.jl index 27ab5a6d98..88582d2ac3 100644 --- a/test/manifolds/invertible_matrices.jl +++ b/test/manifolds/invertible_matrices.jl @@ -27,10 +27,10 @@ using LinearAlgebra, Manifolds, ManifoldsBase, Test @test repr(Mc) == "InvertibleMatrices(3, ℂ)" Mc2 = InvertibleMatrices(3, ℂ; parameter=:field) @test repr(Mc2) == "InvertibleMatrices(3, ℂ; parameter=:field)" - @test manifold_dimension(Mc) == 2*3^2 + @test manifold_dimension(Mc) == 2 * 3^2 @test check_point(Mc, Ac) == nothing @test_throws DomainError is_point(Mc, Bc; error=:error) @test_throws DomainError is_point(Mc, B; error=:error) @test is_point(Mc, A; error=:error) end -end \ No newline at end of file +end From 770377ff29e3312dd379218e0e154edfd4cc8ed6 Mon Sep 17 00:00:00 2001 From: Ronny Bergmann Date: Thu, 24 Oct 2024 06:11:12 +0200 Subject: [PATCH 3/6] Fix two spurious places where I missed to remove the sym reference after copying from there. --- src/manifolds/InvertibleMatrices.jl | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/src/manifolds/InvertibleMatrices.jl b/src/manifolds/InvertibleMatrices.jl index f99a800446..36e52595a2 100644 --- a/src/manifolds/InvertibleMatrices.jl +++ b/src/manifolds/InvertibleMatrices.jl @@ -1,7 +1,7 @@ @doc raw""" InvertibleMatrices{𝔽,T} <: AbstractDecoratorManifold{𝔽} -The [`AbstractManifold`](@extref `ManifoldsBase.AbstractManifold`) ``\operatorname{Sym}(n)`` +The [`AbstractManifold`](@extref `ManifoldsBase.AbstractManifold`) consisting of the real- or complex-valued invertible matrices, that is the set ```math @@ -40,10 +40,8 @@ end check_point(M::InvertibleMatrices{n,𝔽}, p; kwargs...) Check whether `p` is a valid manifold point on the [`InvertibleMatrices`](@ref) `M`, i.e. -whether `p` is a symmetric matrix of size `(n,n)` with values from the corresponding +whether `p` is an invertible matrix of size `(n,n)` with values from the corresponding [`AbstractNumbers`](@extref ManifoldsBase number-system) `𝔽`. - -The tolerance for the symmetry of `p` can be set using `kwargs...`. """ function check_point(M::InvertibleMatrices, p; kwargs...) if det(p) == 0 From 19239a2e3f37c9931da40fcb494d821628cbb9b8 Mon Sep 17 00:00:00 2001 From: Ronny Bergmann Date: Thu, 24 Oct 2024 10:13:50 +0200 Subject: [PATCH 4/6] Increase test cov. --- src/manifolds/InvertibleMatrices.jl | 8 -------- test/manifolds/invertible_matrices.jl | 3 ++- 2 files changed, 2 insertions(+), 9 deletions(-) diff --git a/src/manifolds/InvertibleMatrices.jl b/src/manifolds/InvertibleMatrices.jl index 36e52595a2..a121609cef 100644 --- a/src/manifolds/InvertibleMatrices.jl +++ b/src/manifolds/InvertibleMatrices.jl @@ -28,14 +28,6 @@ function active_traits(f, ::InvertibleMatrices, args...) return merge_traits(IsEmbeddedSubmanifold()) end -function allocation_promotion_function( - ::InvertibleMatrices{ℂ,<:Any}, - ::typeof(get_vector), - args::Tuple, -) - return complex -end - @doc raw""" check_point(M::InvertibleMatrices{n,𝔽}, p; kwargs...) diff --git a/test/manifolds/invertible_matrices.jl b/test/manifolds/invertible_matrices.jl index 88582d2ac3..ebf5840ab6 100644 --- a/test/manifolds/invertible_matrices.jl +++ b/test/manifolds/invertible_matrices.jl @@ -15,8 +15,9 @@ using LinearAlgebra, Manifolds, ManifoldsBase, Test @test_throws DomainError is_point(M, B; error=:error) @test_throws ManifoldDomainError is_point(M, Ac; error=:error) @test_throws ManifoldDomainError is_vector(M, A, Ac; error=:error) + @test is_vector(M, A, A) @test is_flat(M) - @test typeof(get_embedding(M)) === Euclidean{TypeParameter{Tuple{3,3}},ℝ} + @test typeof(get_embedding(M)) === Euclidean{ManifoldsBase.TypeParameter{Tuple{3,3}},ℝ} @test typeof(get_embedding(M2)) === Euclidean{Tuple{Int64,Int64},ℝ} @test embed(M, A) === A @test embed(M, A, A) === A From 79c1888fcd143c7ff17866628f4e8aa2b4e0de07 Mon Sep 17 00:00:00 2001 From: Ronny Bergmann Date: Thu, 24 Oct 2024 10:18:26 +0200 Subject: [PATCH 5/6] runs formatter. --- test/manifolds/invertible_matrices.jl | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/test/manifolds/invertible_matrices.jl b/test/manifolds/invertible_matrices.jl index ebf5840ab6..cb45fc8b54 100644 --- a/test/manifolds/invertible_matrices.jl +++ b/test/manifolds/invertible_matrices.jl @@ -17,7 +17,8 @@ using LinearAlgebra, Manifolds, ManifoldsBase, Test @test_throws ManifoldDomainError is_vector(M, A, Ac; error=:error) @test is_vector(M, A, A) @test is_flat(M) - @test typeof(get_embedding(M)) === Euclidean{ManifoldsBase.TypeParameter{Tuple{3,3}},ℝ} + @test typeof(get_embedding(M)) === + Euclidean{ManifoldsBase.TypeParameter{Tuple{3,3}},ℝ} @test typeof(get_embedding(M2)) === Euclidean{Tuple{Int64,Int64},ℝ} @test embed(M, A) === A @test embed(M, A, A) === A From ce73ac3f8577d568dcbf959cf45d9b2a2d0bff0f Mon Sep 17 00:00:00 2001 From: Ronny Bergmann Date: Thu, 24 Oct 2024 10:41:45 +0200 Subject: [PATCH 6/6] Apply suggestions from code review Co-authored-by: Mateusz Baran --- src/manifolds/InvertibleMatrices.jl | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/manifolds/InvertibleMatrices.jl b/src/manifolds/InvertibleMatrices.jl index a121609cef..c898d3d36f 100644 --- a/src/manifolds/InvertibleMatrices.jl +++ b/src/manifolds/InvertibleMatrices.jl @@ -49,7 +49,7 @@ end check_vector(M::InvertibleMatrices{n,𝔽}, p, X; kwargs... ) Check whether `X` is a tangent vector to manifold point `p` on the -[`InvertibleMatrices`](@ref) `M`, which are all matrces of size ``n×n``` +[`InvertibleMatrices`](@ref) `M`, which are all matrices of size ``n×n`` its values have to be from the correct [`AbstractNumbers`](@extref ManifoldsBase number-system). """ function check_vector(M::InvertibleMatrices, p, X; kwargs...) @@ -78,7 +78,7 @@ is_flat(M::InvertibleMatrices) = true manifold_dimension(M::InvertibleMatrices{n,𝔽}) Return the dimension of the [`InvertibleMatrices`](@ref) matrix `M` over the number system -`𝔽`, which is the same dimension as its embedding, the [`Euclidean`](@ref)`(n,n)`. +`𝔽`, which is the same dimension as its embedding, the [`Euclidean`](@ref)`(n, n; field=𝔽)`. """ function manifold_dimension(M::InvertibleMatrices{<:Any,𝔽}) where {𝔽} return manifold_dimension(get_embedding(M))