Skip to content

Commit a6c47f6

Browse files
committed
feat(properties): add property implications
1 parent 8a909d8 commit a6c47f6

File tree

4 files changed

+92
-5
lines changed

4 files changed

+92
-5
lines changed

src/matrices/companion.jl

+1-1
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,7 @@ Companion{T}(n::Integer) where {T<:Number} = Companion(T[1:n;])
3636
Companion{T}(polynomial::Polynomial) where {T<:Number} = Companion(T.(-polynomial.coeffs[end-1:-1:begin] ./ polynomial.coeffs[end]))
3737

3838
# metadata
39-
@properties Companion Symbol[]
39+
@properties Companion Property[]
4040

4141
# properties
4242
size(A::Companion) = (A.n, A.n)

src/metadata.jl

+67-1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@ const PROPERTIES = Dict{Type{<:PropertyTypes.AbstractProperty},Property}(
1717
PropertyTypes.FixedSize => :fixedsize,
1818
PropertyTypes.Graph => :graph,
1919
PropertyTypes.Hankel => :hankel,
20+
PropertyTypes.Hermitian => :hermitian,
2021
PropertyTypes.Hessenberg => :hessenberg,
2122
PropertyTypes.IllConditioned => :illcond,
2223
PropertyTypes.Indefinite => :indefinite,
@@ -30,6 +31,7 @@ const PROPERTIES = Dict{Type{<:PropertyTypes.AbstractProperty},Property}(
3031
PropertyTypes.Orthogonal => :orthogonal,
3132
PropertyTypes.Positive => :positive,
3233
PropertyTypes.PositiveDefinite => :posdef,
34+
PropertyTypes.PositiveSemidefinite => :possemidef,
3335
PropertyTypes.Random => :random,
3436
PropertyTypes.RankDeficient => :rankdef,
3537
PropertyTypes.Rectangular => :rectangular,
@@ -42,10 +44,73 @@ const PROPERTIES = Dict{Type{<:PropertyTypes.AbstractProperty},Property}(
4244
PropertyTypes.TotallyPositive => :totpos,
4345
PropertyTypes.Triangular => :triangular,
4446
PropertyTypes.Tridiagonal => :tridiagonal,
45-
PropertyTypes.Unimodular => :unimodular
47+
PropertyTypes.Unimodular => :unimodular,
48+
PropertyTypes.Unitary => :unitary
4649
)
4750
)
4851

52+
# Handle property implications
53+
struct PropertyImplication
54+
ifPresent::Set{Property}
55+
ifAbsent::Set{Property}
56+
thenAdd::Set{Property}
57+
end
58+
59+
const IMPLICATIONS = [
60+
PropertyImplication(Set(Property(present)), Set(Property(absent)), Set(Property(toAdd)))
61+
for (present::Vector{Symbol}, absent::Vector{Symbol}, toAdd::Vector{Symbol}) in [
62+
([:bidiagonal], [], [:hessenberg, :sparse, :triangular, :tridiagonal]),
63+
([:circulant], [], [:toeplitz, :eigen]),
64+
([:complex], [], [:normal]),
65+
([:correlation], [], [:symmetric, :possemidef]),
66+
([:diagdom], [], [:posdef]),
67+
([:hankel], [], [:symmetric]),
68+
([:hermitian], [], [:normal]),
69+
([:infdiv], [], [:possemidef]),
70+
([:involutory], [], [:inverse]),
71+
([:involutory, :symmetric], [], [:orthofonal]),
72+
([:involutory, :normal], [:complex], [:symmetric, :orthogonal]),
73+
([:involutory, :normal, :complex], [], [:hermitian, :unitary]),
74+
([:orthogonal], [:complex], [:unitary]),
75+
([:positive], [], [:nonneg]),
76+
([:posdef], [], [:possemidef]),
77+
([:posdef], [:complex], [:symmetric]),
78+
([:posdef], [:complex], [:hermitian]),
79+
([:symmetric], [], [:normal]),
80+
([:symmetric], [:complex], [:hermitian]),
81+
([:TotallyNonnegative], [], [:nonneg]),
82+
([:TotallyNonnegative, :hermitian], [], [:possemidef]),
83+
([:totallyPositive], [], [:positive]),
84+
([:totallyPositive, :hermitian], [], [:posdef]),
85+
([:tridiagonal], [], [:hessenberg]),
86+
([:unimodular], [], [:integer]),
87+
([:unitary], [:complex], [:symmetric])
88+
]
89+
]
90+
91+
"""
92+
add_implied_properties(properties::Set{Symbol})
93+
94+
Add implied properties to a set of properties.
95+
96+
# Examples
97+
```julia-repl
98+
julia> add_implied_properties(Set([:posdef]))
99+
"""
100+
function add_implied_properties!(properties::Vector{Property})
101+
changed = true
102+
while changed
103+
changed = false
104+
for imp in IMPLICATIONS
105+
if imp.ifPresent properties && isempty(imp.ifAbsent properties) && !(imp.thenAdd properties)
106+
append!(properties, setdiff(imp.thenAdd, properties))
107+
changed = true
108+
end
109+
end
110+
end
111+
return properties
112+
end
113+
49114
"""
50115
list_properties()
51116
@@ -154,6 +219,7 @@ julia> register_properties(Matrix, [PropertyTypes.Symmetric(), PropertyTypes.Inv
154219
function register_properties(T::Type, props::Vector{Property})
155220
# check props
156221
check_properties_exist(props...)
222+
add_implied_properties!(props)
157223

158224
# register properties
159225
@eval properties(::Type{<:$T}) = $props

src/types.jl

+18
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,10 @@ The matrix is a Hankel matrix.
6666
"""
6767
struct Hankel <: AbstractProperty end
6868
"""
69+
The matrix is Hermitian.
70+
"""
71+
struct Hermitian <: AbstractProperty end
72+
"""
6973
The matrix is an upper or lower Hessenberg matrix.
7074
"""
7175
struct Hessenberg <: AbstractProperty end
@@ -118,6 +122,10 @@ The matrix is positive definite for some parameter values.
118122
"""
119123
struct PositiveDefinite <: AbstractProperty end
120124
"""
125+
The matrix is positive semidefinite for some parameter values.
126+
"""
127+
struct PositiveSemidefinite <: AbstractProperty end
128+
"""
121129
The matrix has random entries.
122130
"""
123131
struct Random <: AbstractProperty end
@@ -169,6 +177,10 @@ struct TotallyPositive <: AbstractProperty end
169177
The matrix is unimodular for some parameter values.
170178
"""
171179
struct Unimodular <: AbstractProperty end
180+
"""
181+
The matrix is unitary for some parameter values.
182+
"""
183+
struct Unitary <: AbstractProperty end
172184
end
173185

174186
"""
@@ -181,6 +193,12 @@ See also [`list_properties`](@ref), [`@properties`](@ref), [`properties`](@ref).
181193
struct Property
182194
name::Symbol
183195
end
196+
# The alternative solution
197+
# Property(symbols::Symbols...) = [Property(symbol) for symbol in symbols]
198+
# Property(symbols::Vector{Symbol}) = Property(symbols...)
199+
# is not type stable for vectors of length 1.
200+
Property(symbols::Symbol...) = [Property(symbol) for symbol in symbols]
201+
Property(symbols::Vector{Symbol}) = [Property(symbol) for symbol in symbols]
184202

185203
"""
186204
Group

test/metadata.jl

+6-3
Original file line numberDiff line numberDiff line change
@@ -38,9 +38,12 @@ props = list_properties()
3838
@properties Matrix Property[]
3939

4040
# properties
41+
@test Property([:symmetric]) == [Property(:symmetric)]
42+
@test Property(:symmetric, :inverse) == [Property(:symmetric), Property(:inverse)]
4143
@test properties(AbstractMatrix) == []
42-
@properties Matrix [:symmetric, :inverse]
43-
@test properties(Matrix) == [Property(:symmetric), Property(:inverse)]
44-
@test properties(Matrix(ones(1, 1))) == [Property(:symmetric), Property(:inverse)]
44+
@properties Matrix [:posdef, :inverse]
45+
implied_properties = Set([Property(:posdef), Property(:possemidef), Property(:symmetric), Property(:hermitian), Property(:normal), Property(:inverse)])
46+
@test Set(properties(Matrix)) == implied_properties
47+
@test Set(properties(Matrix(ones(1, 1)))) == implied_properties
4548
@properties Matrix Property[]
4649
end

0 commit comments

Comments
 (0)