From 11dd2b17fe5952d0f0992d798bf813bd1939de47 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Fri, 20 Oct 2023 13:57:51 +0530 Subject: [PATCH 01/45] feat!: remove `AbstractArray` subtype --- src/RecursiveArrayTools.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/RecursiveArrayTools.jl b/src/RecursiveArrayTools.jl index 51fa3b0a..1b169acd 100644 --- a/src/RecursiveArrayTools.jl +++ b/src/RecursiveArrayTools.jl @@ -13,7 +13,7 @@ import Adapt import Tables, IteratorInterfaceExtensions -abstract type AbstractVectorOfArray{T, N, A} <: AbstractArray{T, N} end +abstract type AbstractVectorOfArray{T, N, A} end abstract type AbstractDiffEqArray{T, N, A} <: AbstractVectorOfArray{T, N, A} end include("utils.jl") From c6eafa143f004eedb38c3b5a835510cd8484d889 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Mon, 23 Oct 2023 13:46:16 +0530 Subject: [PATCH 02/45] feat: remove issymbollike --- src/vector_of_array.jl | 4 ---- test/utils_test.jl | 4 ---- 2 files changed, 8 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 64126f05..1c165c33 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -64,10 +64,6 @@ end struct AllObserved end -# extended by Symbolcs -issymbollike(::Any) = false -issymbollike(::Union{Symbol, AllObserved}) = true - function Base.Array(VA::AbstractVectorOfArray{T, N, A}) where {T, N, A <: AbstractVector{ <:AbstractVector diff --git a/test/utils_test.jl b/test/utils_test.jl index 1b6b6bdd..01bad869 100644 --- a/test/utils_test.jl +++ b/test/utils_test.jl @@ -67,10 +67,6 @@ function test_recursive_bottom_eltype() end test_recursive_bottom_eltype() -using RecursiveArrayTools: issymbollike -@test !issymbollike(1) -@test issymbollike(:a) - x = zeros(10) recursivefill!(x, 1.0) @test x == ones(10) From 4e3f80d4bba3f202db78f9245cd1f829c948e929 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Mon, 23 Oct 2023 13:47:36 +0530 Subject: [PATCH 03/45] feat: rework DiffEqArray, use new SymbolicIndexingInterface --- Project.toml | 2 +- src/vector_of_array.jl | 139 ++++++++++++++++++++++++----------------- 2 files changed, 82 insertions(+), 59 deletions(-) diff --git a/Project.toml b/Project.toml index b2296d4f..998f2e62 100644 --- a/Project.toml +++ b/Project.toml @@ -39,7 +39,7 @@ RecipesBase = "0.7, 0.8, 1.0" Requires = "1.0" StaticArraysCore = "1.1" Statistics = "1" -SymbolicIndexingInterface = "0.1, 0.2" +SymbolicIndexingInterface = "0.3" Tables = "1" Zygote = "0.6.56" julia = "1.6" diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 1c165c33..e42a67af 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -53,12 +53,11 @@ A[1, :] # all time periods for f(t) A.t ``` """ -mutable struct DiffEqArray{T, N, A, B, C, E, F} <: AbstractDiffEqArray{T, N, A} +mutable struct DiffEqArray{T, N, A, B, F, S} <: AbstractDiffEqArray{T, N, A} u::A # A <: AbstractVector{<: AbstractArray{T, N - 1}} t::B - sc::C - observed::E p::F + sys::S end ### Abstract Interface struct AllObserved @@ -115,36 +114,54 @@ function VectorOfArray(vec::AbstractVector{VT}) where {T, N, VT <: AbstractArray VectorOfArray{T, N + 1, typeof(vec)}(vec) end -function DiffEqArray(vec::AbstractVector{T}, ts, ::NTuple{N, Int}, syms = nothing, - indepsym = nothing, observed = nothing, p = nothing) where {T, N} - sc = if isnothing(indepsym) || indepsym isa AbstractArray - SymbolCache{typeof(syms), typeof(indepsym), Nothing}(syms, indepsym, nothing) - else - SymbolCache{typeof(syms), Vector{typeof(indepsym)}, Nothing}(syms, [indepsym], - nothing) - end - DiffEqArray{eltype(T), N, typeof(vec), typeof(ts), typeof(sc), typeof(observed), - typeof(p)}(vec, ts, sc, observed, p) +function DiffEqArray(vec::AbstractVector{T}, ts, ::NTuple{N, Int}, p = nothing, sys = nothing) where {T, N} + DiffEqArray{eltype(T), N, typeof(vec), typeof(ts), typeof(p), typeof(sys)}(vec, ts, p, sys) end # Assume that the first element is representative of all other elements -function DiffEqArray(vec::AbstractVector, ts::AbstractVector, syms = nothing, - indepsym = nothing, observed = nothing, p = nothing) - DiffEqArray(vec, ts, (size(vec[1])..., length(vec)), syms, indepsym, observed, p) -end -function DiffEqArray(vec::AbstractVector{VT}, ts::AbstractVector, syms = nothing, - indepsym = nothing, observed = nothing, - p = nothing) where {T, N, VT <: AbstractArray{T, N}} - sc = if isnothing(indepsym) || indepsym isa AbstractArray - SymbolCache{typeof(syms), typeof(indepsym), Nothing}(syms, indepsym, nothing) - else - SymbolCache{typeof(syms), Vector{typeof(indepsym)}, Nothing}(syms, [indepsym], - nothing) - end - DiffEqArray{T, N + 1, typeof(vec), typeof(ts), typeof(sc), typeof(observed), typeof(p)}(vec, - ts, - sc, - observed, - p) +function DiffEqArray(vec::AbstractVector, ts::AbstractVector, p = nothing, sys = nothing) + DiffEqArray(vec, ts, (size(vec[1])..., length(vec)), p, sys) +end +function DiffEqArray(vec::AbstractVector{VT}, ts::AbstractVector, p = nothing, sys = nothing) where {T, N, VT <: AbstractArray{T, N}} + DiffEqArray{T, N + 1, typeof(vec), typeof(ts), typeof(p), typeof(sys)}(vec, ts, p, sys) +end + +# SymbolicIndexingInterface implementation for DiffEqArray +# Just forward to A.sys +function SymbolicIndexingInterface.is_variable(A::DiffEqArray, sym) + return is_variable(A.sys, sym) +end +function SymbolicIndexingInterface.has_static_variable(A::DiffEqArray) + return has_static_variable(A.sys) +end +function SymbolicIndexingInterface.variable_index(A::DiffEqArray, sym) + return variable_index(A.sys, sym) +end +function SymbolicIndexingInterface.variable_index(A::DiffEqArray, sym, t) + return variable_index(A.sys, sym, t) +end +function SymbolicIndexingInterface.is_parameter(A::DiffEqArray, sym) + return is_parameter(A.sys, sym) +end +function SymbolicIndexingInterface.parameter_index(A::DiffEqArray, sym) + return parameter_index(A.sys, sym) +end +function SymbolicIndexingInterface.is_independent_variable(A::DiffEqArray, sym) + return is_independent_variable(A.sys, sym) +end +function SymbolicIndexingInterface.is_observed(A::DiffEqArray, sym) + return is_observed(A.sys, sym) +end +function SymbolicIndexingInterface.observed(A::DiffEqArray, sym) + return observed(A.sys, sym) +end +function SymbolicIndexingInterface.observed(A::DiffEqArray, sym, symbolic_states) + return observed(A.sys, sym, symbolic_states) +end +function SymbolicIndexingInterface.is_time_dependent(A::DiffEqArray) + return is_time_dependent(A.sys) +end +function SymbolicIndexingInterface.constant_structure(A::DiffEqArray) + return constant_structure(A.sys) end # Interface for the linear indexing. This is just a view of the underlying nested structure @@ -215,39 +232,45 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, i: end Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, sym) where {T, N} - if issymbollike(sym) && !isnothing(A.sc) - if is_indep_sym(A.sc, sym) - return A.t - elseif is_state_sym(A.sc, sym) - return getindex.(A.u, state_sym_to_index(A.sc, sym)) - elseif is_param_sym(A.sc, sym) - return A.p[param_sym_to_index(A.sc, sym)] - elseif A.observed !== nothing - return observed(A, sym, :) + A.sys === nothing && error("Cannot use symbolic indexing without a system") + + if is_independent_variable(A, sym) + return A.t + elseif is_variable(A, sym) + if has_static_variable(A) + return getindex.(A.u, variable_index(A, sym)) + else + return getindex.(A.u, variable_index.((A, ), (sym, ), A.t)) end - elseif all(issymbollike, sym) && !isnothing(A.sc) - if all(Base.Fix1(is_param_sym, A.sc), sym) + elseif is_parameter(A, sym) + return A.p[parameter_index(A, sym)] + elseif issymbolic(sym) == Symbolic() + return _observed(A, sym, :) + elseif all(isequal(Symbolic()), issymbolic.(collect(sym))) + if all(x -> is_parameter(A, x), collect(sym)) return getindex.((A,), sym) else return [getindex.((A,), sym, i) for i in eachindex(A.t)] end end - return getindex.(A.u, sym) end Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, sym, args...) where {T, N} - if issymbollike(sym) && !isnothing(A.sc) - if is_indep_sym(A.sc, sym) - return A.t[args...] - elseif is_state_sym(A.sc, sym) + A.sys === nothing && error("Cannot use symbolic indexing without a system") + + if is_independent_variable(A, sym) + return A.t[args...] + elseif is_variable(A.sys, sym) + if has_static_variable(A) return A[sym][args...] - elseif A.observed !== nothing - return observed(A, sym, args...) + else + return getindex.(A.u, variable_index.((A, ), (sym, ), A.t[args...])) end - elseif all(issymbollike, sym) && !isnothing(A.sc) - return reduce(vcat, map(s -> A[s, args...]', sym)) + elseif issymbolic(sym) == Symbolic() + return observed(A, sym, args...) + elseif all(isequal(Symbolic()), issymbolic.(collect(sym))) + return reduce(vcat, map(s -> A[s, args...]' ,sym)) end - return getindex.(A.u, sym) end Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, I::Int...) where {T, N} @@ -265,14 +288,14 @@ Base.@propagate_inbounds function Base.getindex(VA::AbstractDiffEqArray{T, N}, return VA.u[i][jj] end -function observed(A::AbstractDiffEqArray{T, N}, sym, i::Int) where {T, N} - A.observed(sym, A.u[i], A.p, A.t[i]) +function _observed(A::AbstractDiffEqArray{T, N}, sym, i::Int) where {T, N} + observed(A, sym)(A.u[i], A.p, A.t[i]) end -function observed(A::AbstractDiffEqArray{T, N}, sym, i::AbstractArray{Int}) where {T, N} - A.observed.((sym,), A.u[i], (A.p,), A.t[i]) +function _observed(A::AbstractDiffEqArray{T, N}, sym, i::AbstractArray{Int}) where {T, N} + observed(A, sym).(A.u[i], (A.p,), A.t[i]) end -function observed(A::AbstractDiffEqArray{T, N}, sym, ::Colon) where {T, N} - A.observed.((sym,), A.u, (A.p,), A.t) +function _observed(A::AbstractDiffEqArray{T, N}, sym, ::Colon) where {T, N} + observed(A, sym).(A.u, (A.p,), A.t) end Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, i::Int, From 0966545c7bf9bb7f39fdc41eff2e8a75af8464f5 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Mon, 23 Oct 2023 13:48:03 +0530 Subject: [PATCH 04/45] fix: add Base.convert method, fix Base.summary --- src/vector_of_array.jl | 11 +++++++++-- 1 file changed, 9 insertions(+), 2 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index e42a67af..14ca60f7 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -423,6 +423,13 @@ end # conversion tools vecarr_to_vectors(VA::AbstractVectorOfArray) = [VA[i, :] for i in eachindex(VA[1])] Base.vec(VA::AbstractVectorOfArray) = vec(convert(Array, VA)) # Allocates +# stack non-ragged arrays to convert them +function Base.convert(::Type{Array}, VA::AbstractVectorOfArray) + if !allequal(size.(VA.u)) + error("Can only convert non-ragged VectorOfArray to Array") + end + return stack(VA.u; dims=1) +end # statistics @inline Base.sum(f, VA::AbstractVectorOfArray) = sum(f, Array(VA)) @@ -443,8 +450,8 @@ end function Base.show(io::IO, m::MIME"text/plain", x::AbstractVectorOfArray) (println(io, summary(x), ':'); show(io, m, x.u)) end -function Base.summary(A::AbstractVectorOfArray) - string("VectorOfArray{", eltype(A), ",", ndims(A), "}") +function Base.summary(A::AbstractVectorOfArray{T,N}) where {T,N} + string("VectorOfArray{", T, ",", N, "}") end function Base.show(io::IO, m::MIME"text/plain", x::AbstractDiffEqArray) From 8ea2d630f67f99e0f44ab9b3a2a05fb7e3c76e8e Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Wed, 25 Oct 2023 16:15:51 +0530 Subject: [PATCH 05/45] feat: make VectorOfArrays broadcastable --- src/vector_of_array.jl | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 14ca60f7..6b33288c 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -498,6 +498,8 @@ end function Broadcast.BroadcastStyle(::Type{<:AbstractVectorOfArray{T, N}}) where {T, N} VectorOfArrayStyle{N}() end +# make vectorofarrays broadcastable so they aren't collected +Broadcast.broadcastable(x::AbstractVectorOfArray) = x @inline function Base.copy(bc::Broadcast.Broadcasted{<:VectorOfArrayStyle}) bc = Broadcast.flatten(bc) From 3c499f5a1fc5fc415d800c71f27c927b0379f2cc Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Wed, 25 Oct 2023 16:16:00 +0530 Subject: [PATCH 06/45] feat: VectorOfArray equality testing --- src/vector_of_array.jl | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 6b33288c..a2d05cd5 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -351,6 +351,10 @@ Base.@propagate_inbounds function Base.setindex!(VA::AbstractVectorOfArray{T, N} VA.u[I[end]][Base.front(I)...] = v end +function Base.:(==)(A::AbstractVectorOfArray, B::AbstractVectorOfArray) + return A.u == B.u +end + # The iterator will be over the subarrays of the container, not the individual elements # unlike an true AbstractArray function Base.iterate(VA::AbstractVectorOfArray, state = 1) From c638aa4a045c952458218c99d00b430e75f78893 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Wed, 25 Oct 2023 19:07:04 +0530 Subject: [PATCH 07/45] fix: fix copy method for DiffEqArray --- src/vector_of_array.jl | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index a2d05cd5..6bc449e4 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -366,9 +366,8 @@ tuples(VA::DiffEqArray) = tuple.(VA.t, VA.u) function Base.copy(VA::AbstractDiffEqArray) typeof(VA)(copy(VA.u), copy(VA.t), - (VA.sc === nothing) ? nothing : copy(VA.sc), - (VA.observed === nothing) ? nothing : copy(VA.observed), - (VA.p === nothing) ? nothing : copy(VA.p)) + (VA.p === nothing) ? nothing : copy(VA.p), + (VA.sys === nothing) ? nothing : copy(VA.sys)) end Base.copy(VA::AbstractVectorOfArray) = typeof(VA)(copy(VA.u)) Base.sizehint!(VA::AbstractVectorOfArray{T, N}, i) where {T, N} = sizehint!(VA.u, i) From 5f722d7e17a4a23db358047a8dc0b424bc29a9ae Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Wed, 25 Oct 2023 19:07:30 +0530 Subject: [PATCH 08/45] feat: equality testing between VectorOfArray and AbstractArray --- src/vector_of_array.jl | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 6bc449e4..ed0c1ae5 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -354,6 +354,10 @@ end function Base.:(==)(A::AbstractVectorOfArray, B::AbstractVectorOfArray) return A.u == B.u end +function Base.:(==)(A::AbstractVectorOfArray, B::AbstractArray) + return A.u == B +end +Base.:(==)(A::AbstractArray, B::AbstractVectorOfArray) = B == A # The iterator will be over the subarrays of the container, not the individual elements # unlike an true AbstractArray From 5e1a3aaa9213031c0b1add8cd28a29ea6569598c Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Wed, 25 Oct 2023 19:07:45 +0530 Subject: [PATCH 09/45] feat: Base.axes for VectorOfArray --- src/vector_of_array.jl | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index ed0c1ae5..1b466141 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -338,6 +338,8 @@ end # Interface for the two-dimensional indexing, a more standard AbstractArray interface @inline Base.size(VA::AbstractVectorOfArray) = (size(VA.u[1])..., length(VA.u)) +Base.axes(VA::AbstractVectorOfArray) = Base.OneTo.(size(VA)) +Base.axes(VA::AbstractVectorOfArray, d::Int) = Base.OneTo(size(VA)[d]) Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, I::Int...) where {T, N} VA.u[I[end]][Base.front(I)...] From a452f629f1b16905899b79cb27c8b10a82722275 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Wed, 25 Oct 2023 19:09:07 +0530 Subject: [PATCH 10/45] feat: fallback getindex for AbstractVectorOfArray --- src/vector_of_array.jl | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 1b466141..073f1dcf 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -218,6 +218,14 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray{T, N}, return A[ntuple(x -> Colon(), ndims(A))...][I, J...] end +Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray, args...) + if last(args) isa Union{Integer,CartesianIndex} + return getindex(A.u[last(args)], Base.front(args)...) + else + return stack(getindex.(A.u[last(args)], tuple.(Base.front(args))...)) + end +end + Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, i::Int, ::Colon) where {T, N} [A.u[j][i] for j in 1:length(A)] From 6f83cd434bdfc6129c837833ebbc56284454d419 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Fri, 27 Oct 2023 13:40:07 +0530 Subject: [PATCH 11/45] feat: add Base.eltype for AbstractVectorOfArray --- src/vector_of_array.jl | 1 + 1 file changed, 1 insertion(+) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 073f1dcf..7e0b6a6d 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -403,6 +403,7 @@ function Base.append!(VA::AbstractVectorOfArray{T, N}, end # Tools for creating similar objects +Base.eltype(::VectorOfArray{T}) where {T} = T @inline function Base.similar(VA::VectorOfArray, ::Type{T} = eltype(VA)) where {T} VectorOfArray([similar(VA[i], T) for i in eachindex(VA)]) end From 6c41391272daa6452521efa830cd1f956170aa40 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Thu, 2 Nov 2023 14:19:26 +0530 Subject: [PATCH 12/45] feat: add AbstractArray methods, fix reverse and convert --- src/vector_of_array.jl | 36 ++++++++++++++++++++++++++++++++++-- 1 file changed, 34 insertions(+), 2 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 7e0b6a6d..feb3f82f 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -388,7 +388,7 @@ Base.sizehint!(VA::AbstractVectorOfArray{T, N}, i) where {T, N} = sizehint!(VA.u Base.reverse!(VA::AbstractVectorOfArray) = reverse!(VA.u) Base.reverse(VA::VectorOfArray) = VectorOfArray(reverse(VA.u)) -Base.reverse(VA::DiffEqArray) = DiffEqArray(reverse(VA.u), VA.t, VA.sc, VA.observed, VA.p) +Base.reverse(VA::DiffEqArray) = DiffEqArray(reverse(VA.u), VA.t, VA.p, VA.sys) function Base.push!(VA::AbstractVectorOfArray{T, N}, new_item::AbstractArray) where {T, N} push!(VA.u, new_item) @@ -402,6 +402,38 @@ function Base.append!(VA::AbstractVectorOfArray{T, N}, return VA end +# AbstractArray methods +Base.ndims(::AbstractVectorOfArray{T,N}) where {T,N} = N +function Base.checkbounds(::Type{Bool}, VA::AbstractVectorOfArray, idx...) + if checkbounds(Bool, VA.u, last(idx)) + if last(idx) isa Integer + return all(checkbounds.(Bool, (VA.u[last(idx)],), Base.front(idx))) + else + return all(checkbounds.(Bool, VA.u[last(idx)], Base.front(idx))) + end + end + return false +end + +# Operations +for op in [:(Base.:-), :(Base.:+)] + @eval function ($op)(A::AbstractVectorOfArray, B::Union{AbstractVectorOfArray,AbstractArray}) + ($op).(A, B) + end + @eval function ($op)(A::AbstractArray, B::AbstractVectorOfArray) + ($op).(A, B) + end +end + +for op in [:(Base.:/), :(Base.:\), :(Base.:*)] + if op !== :(Base.:/) + @eval ($op)(A::Number, B::AbstractVectorOfArray) = ($op).(A, B) + end + if op !== :(Base.:\) + @eval ($op)(A::AbstractVectorOfArray, B::Number) = ($op).(A, B) + end +end + # Tools for creating similar objects Base.eltype(::VectorOfArray{T}) where {T} = T @inline function Base.similar(VA::VectorOfArray, ::Type{T} = eltype(VA)) where {T} @@ -446,7 +478,7 @@ function Base.convert(::Type{Array}, VA::AbstractVectorOfArray) if !allequal(size.(VA.u)) error("Can only convert non-ragged VectorOfArray to Array") end - return stack(VA.u; dims=1) + return stack(VA.u) end # statistics From 84194437772fa0c174d26d024e3cc2aaf1feae2b Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Thu, 2 Nov 2023 15:40:42 +0530 Subject: [PATCH 13/45] fixup! feat: add AbstractArray methods, fix reverse and convert --- src/vector_of_array.jl | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index feb3f82f..cd87e988 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -416,6 +416,14 @@ function Base.checkbounds(::Type{Bool}, VA::AbstractVectorOfArray, idx...) end # Operations +function Base.isapprox(A::AbstractVectorOfArray, B::Union{AbstractVectorOfArray,AbstractArray}; kwargs...) + return all(isapprox.(A, B; kwargs...)) +end + +function Base.isapprox(A::AbstractArray, B::AbstractVectorOfArray; kwargs...) + return all(isapprox.(A, B; kwargs...)) +end + for op in [:(Base.:-), :(Base.:+)] @eval function ($op)(A::AbstractVectorOfArray, B::Union{AbstractVectorOfArray,AbstractArray}) ($op).(A, B) From 67e6429ba3d71b5dbc3ac479611ec7ccc5285c84 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Thu, 2 Nov 2023 19:18:02 +0530 Subject: [PATCH 14/45] fixup! feat: add AbstractArray methods, fix reverse and convert --- src/vector_of_array.jl | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index cd87e988..8619dbf8 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -442,6 +442,13 @@ for op in [:(Base.:/), :(Base.:\), :(Base.:*)] end end +function Base.CartesianIndices(VA::AbstractVectorOfArray) + if !allequal(size.(VA.u)) + error("CartesianIndices only valid for non-ragged arrays") + end + return CartesianIndices((size(VA.u[1])..., length(VA.u))) +end + # Tools for creating similar objects Base.eltype(::VectorOfArray{T}) where {T} = T @inline function Base.similar(VA::VectorOfArray, ::Type{T} = eltype(VA)) where {T} From 01c6af25199a37f62277f812f0399f9ce552f3e5 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Fri, 3 Nov 2023 12:35:39 +0530 Subject: [PATCH 15/45] fixup! feat: add AbstractArray methods, fix reverse and convert --- src/vector_of_array.jl | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 8619dbf8..a1ab394d 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -451,8 +451,11 @@ end # Tools for creating similar objects Base.eltype(::VectorOfArray{T}) where {T} = T -@inline function Base.similar(VA::VectorOfArray, ::Type{T} = eltype(VA)) where {T} - VectorOfArray([similar(VA[i], T) for i in eachindex(VA)]) +@inline function Base.similar(VA::VectorOfArray, dims::NTuple) + similar(VA, eltype(VA), dims) +end +@inline function Base.similar(VA::VectorOfArray, ::Type{T} = eltype(VA), dims = size(VA)) where {T} + VectorOfArray([similar(VA[i], T, Base.front(dims)) for i in 1:last(dims)]) end recursivecopy(VA::VectorOfArray) = VectorOfArray(copy.(VA.u)) From e0eef08c26535b92da30ecf8ad104403bb103efa Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Mon, 6 Nov 2023 15:23:44 +0530 Subject: [PATCH 16/45] feat: update DiffEqArray constructors, support explicit symbols --- src/vector_of_array.jl | 16 +++++++++------- 1 file changed, 9 insertions(+), 7 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index a1ab394d..2e7ac5d4 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -118,21 +118,23 @@ function DiffEqArray(vec::AbstractVector{T}, ts, ::NTuple{N, Int}, p = nothing, DiffEqArray{eltype(T), N, typeof(vec), typeof(ts), typeof(p), typeof(sys)}(vec, ts, p, sys) end # Assume that the first element is representative of all other elements -function DiffEqArray(vec::AbstractVector, ts::AbstractVector, p = nothing, sys = nothing) - DiffEqArray(vec, ts, (size(vec[1])..., length(vec)), p, sys) +function DiffEqArray(vec::AbstractVector, ts::AbstractVector, p::Union{Nothing,AbstractVector} = nothing; variables = nothing, parameters = nothing, independent_variables = nothing) + sys = SymbolCache(something(variables, []), something(parameters, []), something(independent_variables, [])) + _size = size(vec[1]) + return DiffEqArray{eltype(eltype(vec)),length(_size),typeof(vec),typeof(ts),typeof(p),typeof(sys)}(vec, ts, p, sys) end -function DiffEqArray(vec::AbstractVector{VT}, ts::AbstractVector, p = nothing, sys = nothing) where {T, N, VT <: AbstractArray{T, N}} - DiffEqArray{T, N + 1, typeof(vec), typeof(ts), typeof(p), typeof(sys)}(vec, ts, p, sys) + +function DiffEqArray(vec::AbstractVector{VT}, ts::AbstractVector, p::Union{Nothing,AbstractVector} = nothing; variables = nothing, parameters = nothing, independent_variables = nothing) where {T,N,VT<:AbstractArray{T,N}} + sys = SymbolCache(something(variables, []), something(parameters, []), something(independent_variables, [])) + return DiffEqArray{eltype(eltype(vec)),N+1,typeof(vec),typeof(ts),typeof(p),typeof(sys)}(vec, ts, p, sys) end + # SymbolicIndexingInterface implementation for DiffEqArray # Just forward to A.sys function SymbolicIndexingInterface.is_variable(A::DiffEqArray, sym) return is_variable(A.sys, sym) end -function SymbolicIndexingInterface.has_static_variable(A::DiffEqArray) - return has_static_variable(A.sys) -end function SymbolicIndexingInterface.variable_index(A::DiffEqArray, sym) return variable_index(A.sys, sym) end From 249c2b29513d762cbb51d459b1d588631ab9b847 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Mon, 6 Nov 2023 15:24:03 +0530 Subject: [PATCH 17/45] refactor: improve symbolic indexing to support LabelledArrays --- src/vector_of_array.jl | 21 +++++++++++---------- 1 file changed, 11 insertions(+), 10 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 2e7ac5d4..fffbd021 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -242,27 +242,28 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, i: end Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, sym) where {T, N} - A.sys === nothing && error("Cannot use symbolic indexing without a system") - if is_independent_variable(A, sym) return A.t elseif is_variable(A, sym) - if has_static_variable(A) + if constant_structure(A) return getindex.(A.u, variable_index(A, sym)) else - return getindex.(A.u, variable_index.((A, ), (sym, ), A.t)) + return getindex.(A.u, variable_index.((A, ), (sym, ), eachindex(A.t))) end elseif is_parameter(A, sym) return A.p[parameter_index(A, sym)] - elseif issymbolic(sym) == Symbolic() - return _observed(A, sym, :) - elseif all(isequal(Symbolic()), issymbolic.(collect(sym))) + elseif is_observed(A, sym) + return observed(A, sym, :) + elseif symbolic_type(sym) == ArraySymbolic() + return getindex(A, collect(sym)) + elseif sym isa AbstractArray if all(x -> is_parameter(A, x), collect(sym)) return getindex.((A,), sym) else return [getindex.((A,), sym, i) for i in eachindex(A.t)] end end + return getindex.(A.u, sym) end Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, sym, args...) where {T, N} @@ -271,14 +272,14 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, sy if is_independent_variable(A, sym) return A.t[args...] elseif is_variable(A.sys, sym) - if has_static_variable(A) + if constant_structure(A) return A[sym][args...] else return getindex.(A.u, variable_index.((A, ), (sym, ), A.t[args...])) end - elseif issymbolic(sym) == Symbolic() + elseif is_observed(A, sym) return observed(A, sym, args...) - elseif all(isequal(Symbolic()), issymbolic.(collect(sym))) + else return reduce(vcat, map(s -> A[s, args...]' ,sym)) end end From 9501cf762dfa6d80b568e220d65bab063420ac77 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Mon, 6 Nov 2023 15:25:12 +0530 Subject: [PATCH 18/45] fix: remove references to DiffEqArray.sc --- src/tabletraits.jl | 7 ++----- src/vector_of_array.jl | 4 ++-- 2 files changed, 4 insertions(+), 7 deletions(-) diff --git a/src/tabletraits.jl b/src/tabletraits.jl index 71ba06aa..ccae4510 100644 --- a/src/tabletraits.jl +++ b/src/tabletraits.jl @@ -7,16 +7,13 @@ function Tables.rows(A::AbstractDiffEqArray) N = length(A.u[1]) names = [ :timestamp, - (!(A.sc isa SymbolicIndexingInterface.SymbolCache{Nothing, Nothing, Nothing}) ? - (states(A.sc)[i] for i in 1:N) : - (Symbol("value", i) for i in 1:N))..., + (Symbol("value", i) for i in 1:N)..., ] types = Type[eltype(A.t), (eltype(A.u[1]) for _ in 1:N)...] else names = [ :timestamp, - !(A.sc isa SymbolicIndexingInterface.SymbolCache{Nothing, Nothing, Nothing}) ? - states(A.sc)[1] : :value, + :value, ] types = Type[eltype(A.t), VT] end diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index fffbd021..4183f70f 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -534,8 +534,8 @@ end convert(Array, VA) end @recipe function f(VA::AbstractDiffEqArray) - xguide --> ((VA.sc.indepsym !== nothing) ? string(VA.sc.indepsym) : "") - label --> ((VA.sc.syms !== nothing) ? reshape(string.(VA.sc.syms), 1, :) : "") + xguide --> "" + label --> "" VA.t, VA' end @recipe function f(VA::DiffEqArray{T, 1}) where {T} From 2727653432ed30b6d7a6ee354b5ffd4650778102 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Mon, 6 Nov 2023 15:25:31 +0530 Subject: [PATCH 19/45] refactor: refactor tests with new constructors --- test/symbolic_indexing_interface_test.jl | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/test/symbolic_indexing_interface_test.jl b/test/symbolic_indexing_interface_test.jl index ad60bbc8..50b5d96d 100644 --- a/test/symbolic_indexing_interface_test.jl +++ b/test/symbolic_indexing_interface_test.jl @@ -4,14 +4,14 @@ t = 0.0:0.1:1.0 f(x) = 2x f2(x) = 3x -dx = DiffEqArray([[f(x), f2(x)] for x in t], t, [:a, :b], :t) +dx = DiffEqArray([[f(x), f2(x)] for x in t], t; variables = [:a, :b], independent_variables = [:t]) @test dx[:t] == t @test dx[:a] == [f(x) for x in t] @test dx[:b] == [f2(x) for x in t] -dx = DiffEqArray([[f(x), f2(x)] for x in t], t, [:a, :b], [:t]) +dx = DiffEqArray([[f(x), f2(x)] for x in t], t; variables = [:a, :b], independent_variables = [:t]) @test dx[:t] == t -dx = DiffEqArray([[f(x), f2(x)] for x in t], t, [:a, :b]) +dx = DiffEqArray([[f(x), f2(x)] for x in t], t; variables = [:a, :b]) @test_throws Exception dx[nothing] # make sure it isn't storing [nothing] as indepsym ABC = @SLVector (:a, :b, :c); From d05db292d4bdbd2f8aa015556bc89cc3020b18ef Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Tue, 7 Nov 2023 13:31:04 +0530 Subject: [PATCH 20/45] refactor: update DiffEqArray constructors --- src/vector_of_array.jl | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 4183f70f..35a54510 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -118,14 +118,15 @@ function DiffEqArray(vec::AbstractVector{T}, ts, ::NTuple{N, Int}, p = nothing, DiffEqArray{eltype(T), N, typeof(vec), typeof(ts), typeof(p), typeof(sys)}(vec, ts, p, sys) end # Assume that the first element is representative of all other elements -function DiffEqArray(vec::AbstractVector, ts::AbstractVector, p::Union{Nothing,AbstractVector} = nothing; variables = nothing, parameters = nothing, independent_variables = nothing) - sys = SymbolCache(something(variables, []), something(parameters, []), something(independent_variables, [])) + +function DiffEqArray(vec::AbstractVector, ts::AbstractVector, p::Union{Nothing,AbstractVector} = nothing, sys = nothing; variables = nothing, parameters = nothing, independent_variables = nothing) + sys = something(sys, SymbolCache(something(variables, []), something(parameters, []), something(independent_variables, []))) _size = size(vec[1]) return DiffEqArray{eltype(eltype(vec)),length(_size),typeof(vec),typeof(ts),typeof(p),typeof(sys)}(vec, ts, p, sys) end -function DiffEqArray(vec::AbstractVector{VT}, ts::AbstractVector, p::Union{Nothing,AbstractVector} = nothing; variables = nothing, parameters = nothing, independent_variables = nothing) where {T,N,VT<:AbstractArray{T,N}} - sys = SymbolCache(something(variables, []), something(parameters, []), something(independent_variables, [])) +function DiffEqArray(vec::AbstractVector{VT}, ts::AbstractVector, p::Union{Nothing,AbstractVector} = nothing, sys = nothing; variables = nothing, parameters = nothing, independent_variables = nothing) where {T,N,VT<:AbstractArray{T,N}} + sys = something(sys, SymbolCache(something(variables, []), something(parameters, []), something(independent_variables, []))) return DiffEqArray{eltype(eltype(vec)),N+1,typeof(vec),typeof(ts),typeof(p),typeof(sys)}(vec, ts, p, sys) end From 757cfa0bbb005cd97c292223513141ccd1c8eea9 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Tue, 7 Nov 2023 14:10:30 +0530 Subject: [PATCH 21/45] refactor: update to use new SII --- src/vector_of_array.jl | 14 +++++++++++++- 1 file changed, 13 insertions(+), 1 deletion(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 35a54510..f57f5ce7 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -126,7 +126,7 @@ function DiffEqArray(vec::AbstractVector, ts::AbstractVector, p::Union{Nothing,A end function DiffEqArray(vec::AbstractVector{VT}, ts::AbstractVector, p::Union{Nothing,AbstractVector} = nothing, sys = nothing; variables = nothing, parameters = nothing, independent_variables = nothing) where {T,N,VT<:AbstractArray{T,N}} - sys = something(sys, SymbolCache(something(variables, []), something(parameters, []), something(independent_variables, []))) + sys = SymbolCache(something(variables, []), something(parameters, []), something(independent_variables, [])) return DiffEqArray{eltype(eltype(vec)),N+1,typeof(vec),typeof(ts),typeof(p),typeof(sys)}(vec, ts, p, sys) end @@ -142,15 +142,27 @@ end function SymbolicIndexingInterface.variable_index(A::DiffEqArray, sym, t) return variable_index(A.sys, sym, t) end +function SymbolicIndexingInterface.variable_symbols(A::DiffEqArray) + return variable_symbols(A.sys) +end +function SymbolicIndexingInterface.variable_symbols(A::DiffEqArray, i) + return variable_symbols(A.sys, i) +end function SymbolicIndexingInterface.is_parameter(A::DiffEqArray, sym) return is_parameter(A.sys, sym) end function SymbolicIndexingInterface.parameter_index(A::DiffEqArray, sym) return parameter_index(A.sys, sym) end +function SymbolicIndexingInterface.parameter_symbols(A::DiffEqArray) + return parameter_symbols(A.sys) +end function SymbolicIndexingInterface.is_independent_variable(A::DiffEqArray, sym) return is_independent_variable(A.sys, sym) end +function SymbolicIndexingInterface.independent_variable_symbols(A::DiffEqArray) + return independent_variable_symbols(A.sys) +end function SymbolicIndexingInterface.is_observed(A::DiffEqArray, sym) return is_observed(A.sys, sym) end From d6d8536e55dc16bb15e0d5378ad495012c6862b4 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Tue, 7 Nov 2023 16:05:59 +0530 Subject: [PATCH 22/45] fixup! refactor: update DiffEqArray constructors --- src/vector_of_array.jl | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index f57f5ce7..683b6bfa 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -119,13 +119,13 @@ function DiffEqArray(vec::AbstractVector{T}, ts, ::NTuple{N, Int}, p = nothing, end # Assume that the first element is representative of all other elements -function DiffEqArray(vec::AbstractVector, ts::AbstractVector, p::Union{Nothing,AbstractVector} = nothing, sys = nothing; variables = nothing, parameters = nothing, independent_variables = nothing) +function DiffEqArray(vec::AbstractVector, ts::AbstractVector, p = nothing, sys = nothing; variables = nothing, parameters = nothing, independent_variables = nothing) sys = something(sys, SymbolCache(something(variables, []), something(parameters, []), something(independent_variables, []))) _size = size(vec[1]) return DiffEqArray{eltype(eltype(vec)),length(_size),typeof(vec),typeof(ts),typeof(p),typeof(sys)}(vec, ts, p, sys) end -function DiffEqArray(vec::AbstractVector{VT}, ts::AbstractVector, p::Union{Nothing,AbstractVector} = nothing, sys = nothing; variables = nothing, parameters = nothing, independent_variables = nothing) where {T,N,VT<:AbstractArray{T,N}} +function DiffEqArray(vec::AbstractVector{VT}, ts::AbstractVector, p = nothing, sys = nothing; variables = nothing, parameters = nothing, independent_variables = nothing) where {T,N,VT<:AbstractArray{T,N}} sys = SymbolCache(something(variables, []), something(parameters, []), something(independent_variables, [])) return DiffEqArray{eltype(eltype(vec)),N+1,typeof(vec),typeof(ts),typeof(p),typeof(sys)}(vec, ts, p, sys) end From a629cc7e3b4fa35318c7f2c3e5b4ae006a9c4b56 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Tue, 7 Nov 2023 19:08:33 +0530 Subject: [PATCH 23/45] fix: use new SII for tabletraits and plots --- src/tabletraits.jl | 6 ++++-- src/vector_of_array.jl | 6 ++++-- 2 files changed, 8 insertions(+), 4 deletions(-) diff --git a/src/tabletraits.jl b/src/tabletraits.jl index ccae4510..74b65593 100644 --- a/src/tabletraits.jl +++ b/src/tabletraits.jl @@ -7,13 +7,15 @@ function Tables.rows(A::AbstractDiffEqArray) N = length(A.u[1]) names = [ :timestamp, - (Symbol("value", i) for i in 1:N)..., + (isempty(variable_symbols(A)) ? + (Symbol("value", i) for i in 1:N) : + Symbol.(variable_symbols(A)))..., ] types = Type[eltype(A.t), (eltype(A.u[1]) for _ in 1:N)...] else names = [ :timestamp, - :value, + (isempty(variable_symbols(A)) ? :value : Symbol(variable_symbols(A)[1])), ] types = Type[eltype(A.t), VT] end diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 683b6bfa..bd04ebef 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -547,8 +547,10 @@ end convert(Array, VA) end @recipe function f(VA::AbstractDiffEqArray) - xguide --> "" - label --> "" + xguide --> isempty(independent_variable_symbols(VA)) ? "" : + independent_variable_symbols(VA)[1] + label --> isempty(variable_symbols(VA)) ? "" : + reshape(string.(variable_symbols(VA)), 1, :) VA.t, VA' end @recipe function f(VA::DiffEqArray{T, 1}) where {T} From ca31986a0d497dc7a4518c446944200ede53112a Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Tue, 7 Nov 2023 19:09:02 +0530 Subject: [PATCH 24/45] refactor: format --- docs/make.jl | 16 +- ext/RecursiveArrayToolsMeasurementsExt.jl | 4 +- ...siveArrayToolsMonteCarloMeasurementsExt.jl | 11 +- ext/RecursiveArrayToolsTrackerExt.jl | 12 +- ext/RecursiveArrayToolsZygoteExt.jl | 31 +-- src/RecursiveArrayTools.jl | 20 +- src/array_partition.jl | 90 ++++--- src/tabletraits.jl | 4 +- src/utils.jl | 38 +-- src/vector_of_array.jl | 236 ++++++++++++------ test/basic_indexing.jl | 8 +- test/downstream/measurements_and_units.jl | 14 +- test/downstream/symbol_indexing.jl | 18 +- test/partitions_and_static_arrays.jl | 2 +- test/partitions_test.jl | 38 +-- test/runtests.jl | 62 +++-- test/symbolic_indexing_interface_test.jl | 10 +- test/testutils.jl | 2 +- test/upstream.jl | 20 +- test/utils_test.jl | 2 +- 20 files changed, 388 insertions(+), 250 deletions(-) diff --git a/docs/make.jl b/docs/make.jl index eb6f13df..221c6057 100644 --- a/docs/make.jl +++ b/docs/make.jl @@ -6,13 +6,13 @@ cp("./docs/Project.toml", "./docs/src/assets/Project.toml", force = true) include("pages.jl") makedocs(sitename = "RecursiveArrayTools.jl", - authors = "Chris Rackauckas", - modules = [RecursiveArrayTools], - clean = true, doctest = false, linkcheck = true, - warnonly = [:missing_docs], - format = Documenter.HTML(assets = ["assets/favicon.ico"], - canonical = "https://docs.sciml.ai/RecursiveArrayTools/stable/"), - pages = pages) + authors = "Chris Rackauckas", + modules = [RecursiveArrayTools], + clean = true, doctest = false, linkcheck = true, + warnonly = [:missing_docs], + format = Documenter.HTML(assets = ["assets/favicon.ico"], + canonical = "https://docs.sciml.ai/RecursiveArrayTools/stable/"), + pages = pages) deploydocs(repo = "github.com/SciML/RecursiveArrayTools.jl.git"; - push_preview = true) + push_preview = true) diff --git a/ext/RecursiveArrayToolsMeasurementsExt.jl b/ext/RecursiveArrayToolsMeasurementsExt.jl index 50eaaa24..d631655e 100644 --- a/ext/RecursiveArrayToolsMeasurementsExt.jl +++ b/ext/RecursiveArrayToolsMeasurementsExt.jl @@ -4,8 +4,8 @@ import RecursiveArrayTools isdefined(Base, :get_extension) ? (import Measurements) : (import ..Measurements) function RecursiveArrayTools.recursive_unitless_bottom_eltype(a::Type{ - <:Measurements.Measurement - }) + <:Measurements.Measurement, +}) typeof(oneunit(a)) end diff --git a/ext/RecursiveArrayToolsMonteCarloMeasurementsExt.jl b/ext/RecursiveArrayToolsMonteCarloMeasurementsExt.jl index 76825d6d..35250da6 100644 --- a/ext/RecursiveArrayToolsMonteCarloMeasurementsExt.jl +++ b/ext/RecursiveArrayToolsMonteCarloMeasurementsExt.jl @@ -1,15 +1,18 @@ module RecursiveArrayToolsMonteCarloMeasurementsExt import RecursiveArrayTools -isdefined(Base, :get_extension) ? (import MonteCarloMeasurements) : (import ..MonteCarloMeasurements) +isdefined(Base, :get_extension) ? (import MonteCarloMeasurements) : +(import ..MonteCarloMeasurements) function RecursiveArrayTools.recursive_unitless_bottom_eltype(a::Type{ - <:MonteCarloMeasurements.Particles - }) + <:MonteCarloMeasurements.Particles, +}) typeof(one(a)) end -function RecursiveArrayTools.recursive_unitless_eltype(a::Type{<:MonteCarloMeasurements.Particles}) +function RecursiveArrayTools.recursive_unitless_eltype(a::Type{ + <:MonteCarloMeasurements.Particles, +}) typeof(one(a)) end diff --git a/ext/RecursiveArrayToolsTrackerExt.jl b/ext/RecursiveArrayToolsTrackerExt.jl index a06163c2..845aff72 100644 --- a/ext/RecursiveArrayToolsTrackerExt.jl +++ b/ext/RecursiveArrayToolsTrackerExt.jl @@ -4,12 +4,12 @@ import RecursiveArrayTools isdefined(Base, :get_extension) ? (import Tracker) : (import ..Tracker) function RecursiveArrayTools.recursivecopy!(b::AbstractArray{T, N}, - a::AbstractArray{T2, N}) where { - T <: - Tracker.TrackedArray, - T2 <: - Tracker.TrackedArray, - N} + a::AbstractArray{T2, N}) where { + T <: + Tracker.TrackedArray, + T2 <: + Tracker.TrackedArray, + N} @inbounds for i in eachindex(a) b[i] = copy(a[i]) end diff --git a/ext/RecursiveArrayToolsZygoteExt.jl b/ext/RecursiveArrayToolsZygoteExt.jl index 644477fa..9434c96d 100644 --- a/ext/RecursiveArrayToolsZygoteExt.jl +++ b/ext/RecursiveArrayToolsZygoteExt.jl @@ -14,7 +14,7 @@ end ChainRulesCore.ProjectTo(x::VectorOfArray) = ChainRulesCore.ProjectTo{VectorOfArray}() function ChainRulesCore.rrule(T::Type{<:RecursiveArrayTools.GPUArraysCore.AbstractGPUArray}, - xs::AbstractVectorOfArray) + xs::AbstractVectorOfArray) T(xs), ȳ -> (ChainRulesCore.NoTangent(), ȳ) end @@ -28,7 +28,7 @@ end end @adjoint function getindex(VA::AbstractVectorOfArray, - i::Union{BitArray, AbstractArray{Bool}}) + i::Union{BitArray, AbstractArray{Bool}}) function AbstractVectorOfArray_getindex_adjoint(Δ) Δ′ = [(i[j] ? Δ[j] : FillArrays.Fill(zero(eltype(x)), size(x))) for (x, j) in zip(VA.u, 1:length(VA))] @@ -48,7 +48,7 @@ end end @adjoint function getindex(VA::AbstractVectorOfArray, - i::Union{Int, AbstractArray{Int}}) + i::Union{Int, AbstractArray{Int}}) function AbstractVectorOfArray_getindex_adjoint(Δ) Δ′ = [(i[j] ? Δ[j] : FillArrays.Fill(zero(eltype(x)), size(x))) for (x, j) in zip(VA.u, 1:length(VA))] @@ -65,8 +65,8 @@ end end @adjoint function getindex(VA::AbstractVectorOfArray, i::Int, - j::Union{Int, AbstractArray{Int}, CartesianIndex, - Colon, BitArray, AbstractArray{Bool}}...) + j::Union{Int, AbstractArray{Int}, CartesianIndex, + Colon, BitArray, AbstractArray{Bool}}...) function AbstractVectorOfArray_getindex_adjoint(Δ) Δ′ = VectorOfArray([zero(x) for (x, j) in zip(VA.u, 1:length(VA))]) Δ′[i, j...] = Δ @@ -76,11 +76,11 @@ end end @adjoint function ArrayPartition(x::S, - ::Type{Val{copy_x}} = Val{false}) where { - S <: - Tuple, - copy_x - } + ::Type{Val{copy_x}} = Val{false}) where { + S <: + Tuple, + copy_x, +} function ArrayPartition_adjoint(_y) y = Array(_y) starts = vcat(0, cumsum(reduce(vcat, length.(x)))) @@ -93,14 +93,17 @@ end @adjoint function VectorOfArray(u) VectorOfArray(u), - y -> (VectorOfArray([y[ntuple(x -> Colon(), ndims(y) - 1)..., i] - for i in 1:size(y)[end]]),) + y -> begin + (VectorOfArray([y[].u[ntuple(x -> Colon(), ndims(y[].u) - 1)..., i] + for i in 1:size(y[].u)[end]]),) + end end @adjoint function DiffEqArray(u, t) DiffEqArray(u, t), - y -> (DiffEqArray([y[ntuple(x -> Colon(), ndims(y) - 1)..., i] for i in 1:size(y)[end]], - t), nothing) + y -> (DiffEqArray([y[].u[ntuple(x -> Colon(), ndims(y[].u) - 1)..., i] + for i in 1:size(y[].u)[end]], + t), nothing) end @adjoint function literal_getproperty(A::ArrayPartition, ::Val{:x}) diff --git a/src/RecursiveArrayTools.jl b/src/RecursiveArrayTools.jl index 1b169acd..46e47fb0 100644 --- a/src/RecursiveArrayTools.jl +++ b/src/RecursiveArrayTools.jl @@ -6,7 +6,7 @@ module RecursiveArrayTools using DocStringExtensions using RecipesBase, StaticArraysCore, Statistics, - ArrayInterface, LinearAlgebra + ArrayInterface, LinearAlgebra using SymbolicIndexingInterface import Adapt @@ -31,18 +31,24 @@ Base.convert(T::Type{<:GPUArraysCore.AbstractGPUArray}, VA::AbstractVectorOfArra import Requires @static if !isdefined(Base, :get_extension) function __init__() - Requires.@require Measurements="eff96d63-e80a-5855-80a2-b1b0885c5ab7" begin include("../ext/RecursiveArrayToolsMeasurementsExt.jl") end - Requires.@require Tracker="9f7883ad-71c0-57eb-9f7f-b5c9e6d3789c" begin include("../ext/RecursiveArrayToolsTrackerExt.jl") end - Requires.@require Zygote="e88e6eb3-aa80-5325-afca-941959d7151f" begin include("../ext/RecursiveArrayToolsZygoteExt.jl") end + Requires.@require Measurements="eff96d63-e80a-5855-80a2-b1b0885c5ab7" begin + include("../ext/RecursiveArrayToolsMeasurementsExt.jl") + end + Requires.@require Tracker="9f7883ad-71c0-57eb-9f7f-b5c9e6d3789c" begin + include("../ext/RecursiveArrayToolsTrackerExt.jl") + end + Requires.@require Zygote="e88e6eb3-aa80-5325-afca-941959d7151f" begin + include("../ext/RecursiveArrayToolsZygoteExt.jl") + end end end export VectorOfArray, DiffEqArray, AbstractVectorOfArray, AbstractDiffEqArray, - AllObserved, vecarr_to_vectors, tuples + AllObserved, vecarr_to_vectors, tuples export recursivecopy, recursivecopy!, recursivefill!, vecvecapply, copyat_or_push!, - vecvec_to_mat, recursive_one, recursive_mean, recursive_bottom_eltype, - recursive_unitless_bottom_eltype, recursive_unitless_eltype + vecvec_to_mat, recursive_one, recursive_mean, recursive_bottom_eltype, + recursive_unitless_bottom_eltype, recursive_unitless_eltype export ArrayPartition diff --git a/src/array_partition.jl b/src/array_partition.jl index 62cd9948..88d37bd9 100644 --- a/src/array_partition.jl +++ b/src/array_partition.jl @@ -91,10 +91,14 @@ Base.zero(A::ArrayPartition, dims::NTuple{N, Int}) where {N} = zero(A) ## Array Base.Array(A::ArrayPartition) = reduce(vcat, Array.(A.x)) -function Base.Array(VA::AbstractVectorOfArray{T, N, A}) where {T, N, - A <: AbstractVector{ - <:ArrayPartition - }} +function Base.Array(VA::AbstractVectorOfArray{ + T, + N, + A, +}) where {T, N, + A <: AbstractVector{ + <:ArrayPartition, + }} reduce(hcat, Array.(VA.u)) end @@ -115,7 +119,7 @@ Base.ones(A::ArrayPartition, dims::NTuple{N, Int}) where {N} = ones(A) # mutable iff all components of ArrayPartition are mutable @generated function ArrayInterface.ismutable(::Type{<:ArrayPartition{T, S}}) where {T, S - } +} res = all(ArrayInterface.ismutable, S.parameters) return :($res) end @@ -256,7 +260,7 @@ end # workaround for https://github.com/SciML/RecursiveArrayTools.jl/issues/49 function Base._unsafe_getindex(::IndexStyle, A::ArrayPartition, - I::Vararg{Union{Real, AbstractArray}, N}) where {N} + I::Vararg{Union{Real, AbstractArray}, N}) where {N} # This is specifically not inlined to prevent excessive allocations in type unstable code shape = Base.index_shape(I...) dest = similar(A.x[1], shape) @@ -264,7 +268,11 @@ function Base._unsafe_getindex(::IndexStyle, A::ArrayPartition, return dest end -Base._maybe_reshape(::IndexCartesian, A::ArrayPartition, I::Vararg{Union{Real, AbstractArray}, N}) where {N} = Vector(A) +function Base._maybe_reshape(::IndexCartesian, + A::ArrayPartition, + I::Vararg{Union{Real, AbstractArray}, N}) where {N} + Vector(A) +end """ setindex!(A::ArrayPartition, v, i::Int, j...) @@ -321,19 +329,19 @@ end # promotion rules @inline function Broadcast.BroadcastStyle(::ArrayPartitionStyle{AStyle}, - ::ArrayPartitionStyle{BStyle}) where {AStyle, - BStyle} + ::ArrayPartitionStyle{BStyle}) where {AStyle, + BStyle} ArrayPartitionStyle(Broadcast.BroadcastStyle(AStyle(), BStyle())) end function Broadcast.BroadcastStyle(::ArrayPartitionStyle{Style}, - ::Broadcast.DefaultArrayStyle{0}) where { - Style <: - Broadcast.BroadcastStyle - } + ::Broadcast.DefaultArrayStyle{0}) where { + Style <: + Broadcast.BroadcastStyle, +} ArrayPartitionStyle{Style}() end function Broadcast.BroadcastStyle(::ArrayPartitionStyle, - ::Broadcast.DefaultArrayStyle{N}) where {N} + ::Broadcast.DefaultArrayStyle{N}) where {N} Broadcast.DefaultArrayStyle{N}() end @@ -343,11 +351,11 @@ combine_styles(args::Tuple{}) = Broadcast.DefaultArrayStyle{0}() end @inline function combine_styles(args::Tuple{Any, Any}) Broadcast.result_style(Broadcast.BroadcastStyle(args[1]), - Broadcast.BroadcastStyle(args[2])) + Broadcast.BroadcastStyle(args[2])) end @inline function combine_styles(args::Tuple) Broadcast.result_style(Broadcast.BroadcastStyle(args[1]), - combine_styles(Base.tail(args))) + combine_styles(Base.tail(args))) end function Broadcast.BroadcastStyle(::Type{ArrayPartition{T, S}}) where {T, S} @@ -355,9 +363,11 @@ function Broadcast.BroadcastStyle(::Type{ArrayPartition{T, S}}) where {T, S} ArrayPartitionStyle(Style) end -@inline function Base.copy(bc::Broadcast.Broadcasted{ArrayPartitionStyle{Style}}) where { - Style - } +@inline function Base.copy(bc::Broadcast.Broadcasted{ + ArrayPartitionStyle{Style}, +}) where { + Style, +} N = npartitions(bc) @inline function f(i) copy(unpack(bc, i)) @@ -366,9 +376,9 @@ end end @inline function Base.copyto!(dest::ArrayPartition, - bc::Broadcast.Broadcasted{ArrayPartitionStyle{Style}}) where { - Style - } + bc::Broadcast.Broadcasted{ArrayPartitionStyle{Style}}) where { + Style, +} N = npartitions(dest, bc) @inline function f(i) copyto!(dest.x[i], unpack(bc, i)) @@ -401,15 +411,15 @@ _npartitions(args::Tuple{}) = 0 Broadcast.Broadcasted(bc.f, unpack_args(i, bc.args)) end @inline function unpack(bc::Broadcast.Broadcasted{ArrayPartitionStyle{Style}}, - i) where {Style} + i) where {Style} Broadcast.Broadcasted(bc.f, unpack_args(i, bc.args)) end @inline function unpack(bc::Broadcast.Broadcasted{Style}, - i) where {Style <: Broadcast.DefaultArrayStyle} + i) where {Style <: Broadcast.DefaultArrayStyle} Broadcast.Broadcasted{Style}(bc.f, unpack_args(i, bc.args)) end @inline function unpack(bc::Broadcast.Broadcasted{ArrayPartitionStyle{Style}}, - i) where {Style <: Broadcast.DefaultArrayStyle} + i) where {Style <: Broadcast.DefaultArrayStyle} Broadcast.Broadcasted{Style}(bc.f, unpack_args(i, bc.args)) end unpack(x, ::Any) = x @@ -438,11 +448,17 @@ function LinearAlgebra.ldiv!(A::Factorization, b::ArrayPartition) end @static if VERSION >= v"1.9" - function LinearAlgebra.ldiv!(A::LinearAlgebra.SVD{T, Tr, M}, b::ArrayPartition) where {Tr, T, M<:AbstractArray{T}} + function LinearAlgebra.ldiv!(A::LinearAlgebra.SVD{T, Tr, M}, + b::ArrayPartition) where {Tr, T, M <: AbstractArray{T}} (x = ldiv!(A, Array(b)); copyto!(b, x)) end - function LinearAlgebra.ldiv!(A::LinearAlgebra.QRCompactWY{T, M, C}, b::ArrayPartition) where {T<:Union{Float32, Float64, ComplexF64, ComplexF32}, M<:AbstractMatrix{T}, C<:AbstractMatrix{T}} + function LinearAlgebra.ldiv!(A::LinearAlgebra.QRCompactWY{T, M, C}, + b::ArrayPartition) where { + T <: Union{Float32, Float64, ComplexF64, ComplexF32}, + M <: AbstractMatrix{T}, + C <: AbstractMatrix{T}, + } (x = ldiv!(A, Array(b)); copyto!(b, x)) end end @@ -534,7 +550,7 @@ function _ldiv!(A::LowerTriangular, bb::ArrayPartition) end function LinearAlgebra.ldiv!(A::LowerTriangular{T, <:LinearAlgebra.Adjoint{T}}, - bb::ArrayPartition) where {T} + bb::ArrayPartition) where {T} _ldiv!(A, bb) end LinearAlgebra.ldiv!(A::LowerTriangular, bb::ArrayPartition) = _ldiv!(A, bb) @@ -580,17 +596,19 @@ function LinearAlgebra.mul!(C::ArrayPartition, A::ArrayPartition, B::ArrayPartit end function Base.convert(::Type{ArrayPartition{T, S}}, - A::ArrayPartition{<:Any, <:NTuple{N, Any}}) where {N, T, - S <: - NTuple{N, Any}} + A::ArrayPartition{<:Any, <:NTuple{N, Any}}) where {N, T, + S <: + NTuple{N, Any}} return ArrayPartition{T, S}(ntuple((@inline i -> convert(S.parameters[i], A.x[i])), - Val(N))) + Val(N))) end -@generated function Base.length(::Type{<:ArrayPartition{F, T}}) where {F, N, - T <: NTuple{N, - StaticArraysCore.StaticArray - }} +@generated function Base.length(::Type{ + <:ArrayPartition{F, T}, +}) where {F, N, + T <: NTuple{N, + StaticArraysCore.StaticArray, + }} sum_expr = Expr(:call, :+) for param in T.parameters push!(sum_expr.args, :(length($param))) diff --git a/src/tabletraits.jl b/src/tabletraits.jl index 74b65593..cd6e9dd8 100644 --- a/src/tabletraits.jl +++ b/src/tabletraits.jl @@ -36,7 +36,7 @@ struct AbstractDiffEqArrayRows{T, U} end function AbstractDiffEqArrayRows(names, types, t, u) AbstractDiffEqArrayRows(Symbol.(names), types, - Dict(Symbol(nm) => i for (i, nm) in enumerate(names)), t, u) + Dict(Symbol(nm) => i for (i, nm) in enumerate(names)), t, u) end Base.length(x::AbstractDiffEqArrayRows) = length(x.u) @@ -44,7 +44,7 @@ function Base.eltype(::Type{AbstractDiffEqArrayRows{T, U}}) where {T, U} AbstractDiffEqArrayRow{eltype(T), eltype(U)} end function Base.iterate(x::AbstractDiffEqArrayRows, - (t_state, u_state) = (iterate(x.t), iterate(x.u))) + (t_state, u_state) = (iterate(x.t), iterate(x.u))) t_state === nothing && return nothing u_state === nothing && return nothing t, _t_state = t_state diff --git a/src/utils.jl b/src/utils.jl index bc8d7951..18dad100 100644 --- a/src/utils.jl +++ b/src/utils.jl @@ -10,7 +10,7 @@ function recursivecopy(a) deepcopy(a) end function recursivecopy(a::Union{StaticArraysCore.SVector, StaticArraysCore.SMatrix, - StaticArraysCore.SArray, Number}) + StaticArraysCore.SArray, Number}) copy(a) end function recursivecopy(a::AbstractArray{T, N}) where {T <: Number, N} @@ -37,9 +37,9 @@ like `copy!` on arrays of scalars. function recursivecopy! end function recursivecopy!(b::AbstractArray{T, N}, - a::AbstractArray{T2, N}) where {T <: StaticArraysCore.StaticArray, - T2 <: StaticArraysCore.StaticArray, - N} + a::AbstractArray{T2, N}) where {T <: StaticArraysCore.StaticArray, + T2 <: StaticArraysCore.StaticArray, + N} @inbounds for i in eachindex(a) # TODO: Check for `setindex!`` and use `copy!(b[i],a[i])` or `b[i] = a[i]`, see #19 b[i] = copy(a[i]) @@ -47,18 +47,18 @@ function recursivecopy!(b::AbstractArray{T, N}, end function recursivecopy!(b::AbstractArray{T, N}, - a::AbstractArray{T2, N}) where {T <: Enum, T2 <: Enum, N} + a::AbstractArray{T2, N}) where {T <: Enum, T2 <: Enum, N} copyto!(b, a) end function recursivecopy!(b::AbstractArray{T, N}, - a::AbstractArray{T2, N}) where {T <: Number, T2 <: Number, N} + a::AbstractArray{T2, N}) where {T <: Number, T2 <: Number, N} copyto!(b, a) end function recursivecopy!(b::AbstractArray{T, N}, - a::AbstractArray{T2, N}) where {T <: AbstractArray, - T2 <: AbstractArray, N} + a::AbstractArray{T2, N}) where {T <: AbstractArray, + T2 <: AbstractArray, N} if ArrayInterface.ismutable(T) @inbounds for i in eachindex(b, a) recursivecopy!(b[i], a[i]) @@ -79,32 +79,32 @@ A recursive `fill!` function. function recursivefill! end function recursivefill!(b::AbstractArray{T, N}, - a::T2) where {T <: StaticArraysCore.StaticArray, - T2 <: StaticArraysCore.StaticArray, N} + a::T2) where {T <: StaticArraysCore.StaticArray, + T2 <: StaticArraysCore.StaticArray, N} @inbounds for i in eachindex(b) b[i] = copy(a) end end function recursivefill!(bs::AbstractVectorOfArray{T, N}, - a::T2) where {T <: StaticArraysCore.StaticArray, - T2 <: StaticArraysCore.StaticArray, N} + a::T2) where {T <: StaticArraysCore.StaticArray, + T2 <: StaticArraysCore.StaticArray, N} @inbounds for b in bs, i in eachindex(b) b[i] = copy(a) end end function recursivefill!(b::AbstractArray{T, N}, - a::T2) where {T <: StaticArraysCore.SArray, - T2 <: Union{Number, Bool}, N} + a::T2) where {T <: StaticArraysCore.SArray, + T2 <: Union{Number, Bool}, N} @inbounds for i in eachindex(b) b[i] = fill(a, typeof(b[i])) end end function recursivefill!(bs::AbstractVectorOfArray{T, N}, - a::T2) where {T <: StaticArraysCore.SArray, - T2 <: Union{Number, Bool}, N} + a::T2) where {T <: StaticArraysCore.SArray, + T2 <: Union{Number, Bool}, N} @inbounds for b in bs, i in eachindex(b) b[i] = fill(a, typeof(b[i])) end @@ -115,8 +115,8 @@ function recursivefill!(b::AbstractArray{T, N}, a::T2) where {T <: Enum, T2 <: E end function recursivefill!(b::AbstractArray{T, N}, - a::T2) where {T <: Union{Number, Bool}, T2 <: Union{Number, Bool}, N - } + a::T2) where {T <: Union{Number, Bool}, T2 <: Union{Number, Bool}, N +} fill!(b, a) end @@ -204,7 +204,7 @@ function copyat_or_push!(a::AbstractVector{T}, i::Int, x, perform_copy = true) w end function copyat_or_push!(a::AbstractVector{T}, i::Int, x, - nc::Type{Val{perform_copy}}) where {T, perform_copy} + nc::Type{Val{perform_copy}}) where {T, perform_copy} copyat_or_push!(a, i, x, perform_copy) end diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index bd04ebef..f6542e64 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -63,37 +63,61 @@ end struct AllObserved end -function Base.Array(VA::AbstractVectorOfArray{T, N, A}) where {T, N, - A <: AbstractVector{ - <:AbstractVector - }} +function Base.Array(VA::AbstractVectorOfArray{ + T, + N, + A, +}) where {T, N, + A <: AbstractVector{ + <:AbstractVector, + }} reduce(hcat, VA.u) end -function Base.Array(VA::AbstractVectorOfArray{T, N, A}) where {T, N, - A <: - AbstractVector{<:Number}} +function Base.Array(VA::AbstractVectorOfArray{ + T, + N, + A, +}) where {T, N, + A <: + AbstractVector{<:Number}} VA.u end -function Base.Matrix(VA::AbstractVectorOfArray{T, N, A}) where {T, N, - A <: AbstractVector{ - <:AbstractVector - }} +function Base.Matrix(VA::AbstractVectorOfArray{ + T, + N, + A, +}) where {T, N, + A <: AbstractVector{ + <:AbstractVector, + }} reduce(hcat, VA.u) end -function Base.Matrix(VA::AbstractVectorOfArray{T, N, A}) where {T, N, - A <: - AbstractVector{<:Number}} +function Base.Matrix(VA::AbstractVectorOfArray{ + T, + N, + A, +}) where {T, N, + A <: + AbstractVector{<:Number}} Matrix(VA.u) end -function Base.Vector(VA::AbstractVectorOfArray{T, N, A}) where {T, N, - A <: AbstractVector{ - <:AbstractVector - }} +function Base.Vector(VA::AbstractVectorOfArray{ + T, + N, + A, +}) where {T, N, + A <: AbstractVector{ + <:AbstractVector, + }} vec(reduce(hcat, VA.u)) end -function Base.Vector(VA::AbstractVectorOfArray{T, N, A}) where {T, N, - A <: - AbstractVector{<:Number}} +function Base.Vector(VA::AbstractVectorOfArray{ + T, + N, + A, +}) where {T, N, + A <: + AbstractVector{<:Number}} VA.u end function Base.Array(VA::AbstractVectorOfArray) @@ -114,23 +138,66 @@ function VectorOfArray(vec::AbstractVector{VT}) where {T, N, VT <: AbstractArray VectorOfArray{T, N + 1, typeof(vec)}(vec) end -function DiffEqArray(vec::AbstractVector{T}, ts, ::NTuple{N, Int}, p = nothing, sys = nothing) where {T, N} - DiffEqArray{eltype(T), N, typeof(vec), typeof(ts), typeof(p), typeof(sys)}(vec, ts, p, sys) +function DiffEqArray(vec::AbstractVector{T}, + ts, + ::NTuple{N, Int}, + p = nothing, + sys = nothing) where {T, N} + DiffEqArray{eltype(T), N, typeof(vec), typeof(ts), typeof(p), typeof(sys)}(vec, + ts, + p, + sys) end # Assume that the first element is representative of all other elements -function DiffEqArray(vec::AbstractVector, ts::AbstractVector, p = nothing, sys = nothing; variables = nothing, parameters = nothing, independent_variables = nothing) - sys = something(sys, SymbolCache(something(variables, []), something(parameters, []), something(independent_variables, []))) +function DiffEqArray(vec::AbstractVector, + ts::AbstractVector, + p = nothing, + sys = nothing; + variables = nothing, + parameters = nothing, + independent_variables = nothing) + sys = something(sys, + SymbolCache(something(variables, []), + something(parameters, []), + something(independent_variables, []))) _size = size(vec[1]) - return DiffEqArray{eltype(eltype(vec)),length(_size),typeof(vec),typeof(ts),typeof(p),typeof(sys)}(vec, ts, p, sys) + return DiffEqArray{ + eltype(eltype(vec)), + length(_size), + typeof(vec), + typeof(ts), + typeof(p), + typeof(sys), + }(vec, + ts, + p, + sys) +end + +function DiffEqArray(vec::AbstractVector{VT}, + ts::AbstractVector, + p = nothing, + sys = nothing; + variables = nothing, + parameters = nothing, + independent_variables = nothing) where {T, N, VT <: AbstractArray{T, N}} + sys = SymbolCache(something(variables, []), + something(parameters, []), + something(independent_variables, [])) + return DiffEqArray{ + eltype(eltype(vec)), + N + 1, + typeof(vec), + typeof(ts), + typeof(p), + typeof(sys), + }(vec, + ts, + p, + sys) end -function DiffEqArray(vec::AbstractVector{VT}, ts::AbstractVector, p = nothing, sys = nothing; variables = nothing, parameters = nothing, independent_variables = nothing) where {T,N,VT<:AbstractArray{T,N}} - sys = SymbolCache(something(variables, []), something(parameters, []), something(independent_variables, [])) - return DiffEqArray{eltype(eltype(vec)),N+1,typeof(vec),typeof(ts),typeof(p),typeof(sys)}(vec, ts, p, sys) -end - - # SymbolicIndexingInterface implementation for DiffEqArray # Just forward to A.sys function SymbolicIndexingInterface.is_variable(A::DiffEqArray, sym) @@ -189,52 +256,52 @@ end # Linear indexing will be over the container elements, not the individual elements # unlike an true AbstractArray Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, - I::Int) where {T, N} + I::Int) where {T, N} VA.u[I] end Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, - I::Colon) where {T, N} + I::Colon) where {T, N} VA.u[I] end Base.@propagate_inbounds function Base.getindex(VA::AbstractDiffEqArray{T, N}, - I::Colon) where {T, N} + I::Colon) where {T, N} VA.u[I] end Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, - I::AbstractArray{Int}) where {T, N} + I::AbstractArray{Int}) where {T, N} VectorOfArray(VA.u[I]) end Base.@propagate_inbounds function Base.getindex(VA::AbstractDiffEqArray{T, N}, - I::AbstractArray{Int}) where {T, N} + I::AbstractArray{Int}) where {T, N} DiffEqArray(VA.u[I], VA.t[I]) end Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, - I::Union{Int, AbstractArray{Int}, - CartesianIndex, Colon, BitArray, - AbstractArray{Bool}}...) where {T, - N} + I::Union{Int, AbstractArray{Int}, + CartesianIndex, Colon, BitArray, + AbstractArray{Bool}}...) where {T, + N} RecursiveArrayTools.VectorOfArray(A.u)[I...] end __parameterless_type(T) = Base.typename(T).wrapper Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray{T, N}, - I::Colon...) where {T, N} + I::Colon...) where {T, N} @assert length(I) == ndims(A.u[1]) + 1 vecs = vec.(A.u) return Adapt.adapt(__parameterless_type(T), - reshape(reduce(hcat, vecs), size(A.u[1])..., length(A.u))) + reshape(reduce(hcat, vecs), size(A.u[1])..., length(A.u))) end Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray{T, N}, - I::AbstractArray{Bool}, - J::Colon...) where {T, N} + I::AbstractArray{Bool}, + J::Colon...) where {T, N} @assert length(J) == ndims(A.u[1]) + 1 - ndims(I) @assert size(I) == size(A)[1:(ndims(A) - length(J))] return A[ntuple(x -> Colon(), ndims(A))...][I, J...] end Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray, args...) - if last(args) isa Union{Integer,CartesianIndex} + if last(args) isa Union{Integer, CartesianIndex} return getindex(A.u[last(args)], Base.front(args)...) else return stack(getindex.(A.u[last(args)], tuple.(Base.front(args))...)) @@ -242,26 +309,26 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray, args.. end Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, i::Int, - ::Colon) where {T, N} + ::Colon) where {T, N} [A.u[j][i] for j in 1:length(A)] end Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, ::Colon, - i::Int) where {T, N} + i::Int) where {T, N} A.u[i] end Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, i::Int, - II::AbstractArray{Int}) where {T, N} + II::AbstractArray{Int}) where {T, N} [A.u[j][i] for j in II] end Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, - sym) where {T, N} + sym) where {T, N} if is_independent_variable(A, sym) return A.t elseif is_variable(A, sym) if constant_structure(A) return getindex.(A.u, variable_index(A, sym)) else - return getindex.(A.u, variable_index.((A, ), (sym, ), eachindex(A.t))) + return getindex.(A.u, variable_index.((A,), (sym,), eachindex(A.t))) end elseif is_parameter(A, sym) return A.p[parameter_index(A, sym)] @@ -279,7 +346,7 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, return getindex.(A.u, sym) end Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, sym, - args...) where {T, N} + args...) where {T, N} A.sys === nothing && error("Cannot use symbolic indexing without a system") if is_independent_variable(A, sym) @@ -288,24 +355,24 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, sy if constant_structure(A) return A[sym][args...] else - return getindex.(A.u, variable_index.((A, ), (sym, ), A.t[args...])) + return getindex.(A.u, variable_index.((A,), (sym,), A.t[args...])) end elseif is_observed(A, sym) return observed(A, sym, args...) else - return reduce(vcat, map(s -> A[s, args...]' ,sym)) + return reduce(vcat, map(s -> A[s, args...]', sym)) end end Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, - I::Int...) where {T, N} + I::Int...) where {T, N} A.u[I[end]][Base.front(I)...] end Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, - i::Int) where {T, N} + i::Int) where {T, N} A.u[i] end Base.@propagate_inbounds function Base.getindex(VA::AbstractDiffEqArray{T, N}, - ii::CartesianIndex) where {T, N} + ii::CartesianIndex) where {T, N} ti = Tuple(ii) i = last(ti) jj = CartesianIndex(Base.front(ti)) @@ -323,37 +390,37 @@ function _observed(A::AbstractDiffEqArray{T, N}, sym, ::Colon) where {T, N} end Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, i::Int, - ::Colon) where {T, N} + ::Colon) where {T, N} [VA.u[j][i] for j in 1:length(VA)] end Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, - ii::CartesianIndex) where {T, N} + ii::CartesianIndex) where {T, N} ti = Tuple(ii) i = last(ti) jj = CartesianIndex(Base.front(ti)) return VA.u[i][jj] end Base.@propagate_inbounds function Base.setindex!(VA::AbstractVectorOfArray{T, N}, v, - I::Int) where {T, N} + I::Int) where {T, N} VA.u[I] = v end Base.@propagate_inbounds function Base.setindex!(VA::AbstractVectorOfArray{T, N}, v, - I::Colon) where {T, N} + I::Colon) where {T, N} VA.u[I] = v end Base.@propagate_inbounds function Base.setindex!(VA::AbstractVectorOfArray{T, N}, v, - I::AbstractArray{Int}) where {T, N} + I::AbstractArray{Int}) where {T, N} VA.u[I] = v end Base.@propagate_inbounds function Base.setindex!(VA::AbstractVectorOfArray{T, N}, v, i::Int, - ::Colon) where {T, N} + ::Colon) where {T, N} for j in 1:length(VA) VA.u[j][i] = v[j] end return v end Base.@propagate_inbounds function Base.setindex!(VA::AbstractVectorOfArray{T, N}, x, - ii::CartesianIndex) where {T, N} + ii::CartesianIndex) where {T, N} ti = Tuple(ii) i = last(ti) jj = CartesianIndex(Base.front(ti)) @@ -365,15 +432,15 @@ end Base.axes(VA::AbstractVectorOfArray) = Base.OneTo.(size(VA)) Base.axes(VA::AbstractVectorOfArray, d::Int) = Base.OneTo(size(VA)[d]) Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, - I::Int...) where {T, N} + I::Int...) where {T, N} VA.u[I[end]][Base.front(I)...] end Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, ::Colon, - I::Int) where {T, N} + I::Int) where {T, N} VA.u[I] end Base.@propagate_inbounds function Base.setindex!(VA::AbstractVectorOfArray{T, N}, v, - I::Int...) where {T, N} + I::Int...) where {T, N} VA.u[I[end]][Base.front(I)...] = v end @@ -395,9 +462,9 @@ tuples(VA::DiffEqArray) = tuple.(VA.t, VA.u) # Growing the array simply adds to the container vector function Base.copy(VA::AbstractDiffEqArray) typeof(VA)(copy(VA.u), - copy(VA.t), - (VA.p === nothing) ? nothing : copy(VA.p), - (VA.sys === nothing) ? nothing : copy(VA.sys)) + copy(VA.t), + (VA.p === nothing) ? nothing : copy(VA.p), + (VA.sys === nothing) ? nothing : copy(VA.sys)) end Base.copy(VA::AbstractVectorOfArray) = typeof(VA)(copy(VA.u)) Base.sizehint!(VA::AbstractVectorOfArray{T, N}, i) where {T, N} = sizehint!(VA.u, i) @@ -411,7 +478,7 @@ function Base.push!(VA::AbstractVectorOfArray{T, N}, new_item::AbstractArray) wh end function Base.append!(VA::AbstractVectorOfArray{T, N}, - new_item::AbstractVectorOfArray{T, N}) where {T, N} + new_item::AbstractVectorOfArray{T, N}) where {T, N} for item in copy(new_item) push!(VA, item) end @@ -419,7 +486,7 @@ function Base.append!(VA::AbstractVectorOfArray{T, N}, end # AbstractArray methods -Base.ndims(::AbstractVectorOfArray{T,N}) where {T,N} = N +Base.ndims(::AbstractVectorOfArray{T, N}) where {T, N} = N function Base.checkbounds(::Type{Bool}, VA::AbstractVectorOfArray, idx...) if checkbounds(Bool, VA.u, last(idx)) if last(idx) isa Integer @@ -432,7 +499,9 @@ function Base.checkbounds(::Type{Bool}, VA::AbstractVectorOfArray, idx...) end # Operations -function Base.isapprox(A::AbstractVectorOfArray, B::Union{AbstractVectorOfArray,AbstractArray}; kwargs...) +function Base.isapprox(A::AbstractVectorOfArray, + B::Union{AbstractVectorOfArray, AbstractArray}; + kwargs...) return all(isapprox.(A, B; kwargs...)) end @@ -441,7 +510,8 @@ function Base.isapprox(A::AbstractArray, B::AbstractVectorOfArray; kwargs...) end for op in [:(Base.:-), :(Base.:+)] - @eval function ($op)(A::AbstractVectorOfArray, B::Union{AbstractVectorOfArray,AbstractArray}) + @eval function ($op)(A::AbstractVectorOfArray, + B::Union{AbstractVectorOfArray, AbstractArray}) ($op).(A, B) end @eval function ($op)(A::AbstractArray, B::AbstractVectorOfArray) @@ -470,7 +540,9 @@ Base.eltype(::VectorOfArray{T}) where {T} = T @inline function Base.similar(VA::VectorOfArray, dims::NTuple) similar(VA, eltype(VA), dims) end -@inline function Base.similar(VA::VectorOfArray, ::Type{T} = eltype(VA), dims = size(VA)) where {T} +@inline function Base.similar(VA::VectorOfArray, + ::Type{T} = eltype(VA), + dims = size(VA)) where {T} VectorOfArray([similar(VA[i], T, Base.front(dims)) for i in 1:last(dims)]) end recursivecopy(VA::VectorOfArray) = VectorOfArray(copy.(VA.u)) @@ -534,7 +606,7 @@ end function Base.show(io::IO, m::MIME"text/plain", x::AbstractVectorOfArray) (println(io, summary(x), ':'); show(io, m, x.u)) end -function Base.summary(A::AbstractVectorOfArray{T,N}) where {T,N} +function Base.summary(A::AbstractVectorOfArray{T, N}) where {T, N} string("VectorOfArray{", T, ",", N, "}") end @@ -570,15 +642,15 @@ VectorOfArrayStyle(::Val{N}) where {N} = VectorOfArrayStyle{N}() # The order is important here. We want to override Base.Broadcast.DefaultArrayStyle to return another Base.Broadcast.DefaultArrayStyle. Broadcast.BroadcastStyle(a::VectorOfArrayStyle, ::Base.Broadcast.DefaultArrayStyle{0}) = a function Broadcast.BroadcastStyle(::VectorOfArrayStyle{N}, - a::Base.Broadcast.DefaultArrayStyle{M}) where {M, N} + a::Base.Broadcast.DefaultArrayStyle{M}) where {M, N} Base.Broadcast.DefaultArrayStyle(Val(max(M, N))) end function Broadcast.BroadcastStyle(::VectorOfArrayStyle{N}, - a::Base.Broadcast.AbstractArrayStyle{M}) where {M, N} + a::Base.Broadcast.AbstractArrayStyle{M}) where {M, N} typeof(a)(Val(max(M, N))) end function Broadcast.BroadcastStyle(::VectorOfArrayStyle{M}, - ::VectorOfArrayStyle{N}) where {M, N} + ::VectorOfArrayStyle{N}) where {M, N} VectorOfArrayStyle(Val(max(M, N))) end function Broadcast.BroadcastStyle(::Type{<:AbstractVectorOfArray{T, N}}) where {T, N} @@ -591,12 +663,12 @@ Broadcast.broadcastable(x::AbstractVectorOfArray) = x bc = Broadcast.flatten(bc) N = narrays(bc) VectorOfArray(map(1:N) do i - copy(unpack_voa(bc, i)) - end) + copy(unpack_voa(bc, i)) + end) end @inline function Base.copyto!(dest::AbstractVectorOfArray, - bc::Broadcast.Broadcasted{<:VectorOfArrayStyle}) + bc::Broadcast.Broadcasted{<:VectorOfArrayStyle}) bc = Broadcast.flatten(bc) N = narrays(bc) @inbounds for i in 1:N diff --git a/test/basic_indexing.jl b/test/basic_indexing.jl index 5ebd7e1e..c69d6a85 100644 --- a/test/basic_indexing.jl +++ b/test/basic_indexing.jl @@ -16,8 +16,8 @@ mulX .= sqrt.(abs.(testva .* X)) @test mulX == ref @test Array(testva) == [1 4 7 - 2 5 8 - 3 6 9] + 2 5 8 + 3 6 9] @test testa[1:2, 1:2] == [1 4; 2 5] @test testva[1:2, 1:2] == [1 4; 2 5] @@ -26,8 +26,8 @@ mulX .= sqrt.(abs.(testva .* X)) t = [1, 2, 3] diffeq = DiffEqArray(recs, t) @test Array(diffeq) == [1 4 7 - 2 5 8 - 3 6 9] + 2 5 8 + 3 6 9] @test diffeq[1:2, 1:2] == [1 4; 2 5] # # ndims == 2 diff --git a/test/downstream/measurements_and_units.jl b/test/downstream/measurements_and_units.jl index a6a39b60..ded9184e 100644 --- a/test/downstream/measurements_and_units.jl +++ b/test/downstream/measurements_and_units.jl @@ -1,9 +1,9 @@ -using Unitful,MonteCarloMeasurements,OrdinaryDiffEq +using Unitful, MonteCarloMeasurements, OrdinaryDiffEq g3 = 9.81u"m/s^2" -du4 = [10.0±.1,10.0±.1].*u"m/s" -tspan3 = (0.0,1.0).*u"s" -f3(du,u,p,t) = [0.0u"m/s^2",-g3] -u3 = [0.0,0.0].*u"m" -problem4 = SecondOrderODEProblem(f3,du4,u3,tspan3) -solve(problem4, Tsit5()) \ No newline at end of file +du4 = [10.0 ± 0.1, 10.0 ± 0.1] .* u"m/s" +tspan3 = (0.0, 1.0) .* u"s" +f3(du, u, p, t) = [0.0u"m/s^2", -g3] +u3 = [0.0, 0.0] .* u"m" +problem4 = SecondOrderODEProblem(f3, du4, u3, tspan3) +solve(problem4, Tsit5()) diff --git a/test/downstream/symbol_indexing.jl b/test/downstream/symbol_indexing.jl index 8110cf7c..eb3739da 100644 --- a/test/downstream/symbol_indexing.jl +++ b/test/downstream/symbol_indexing.jl @@ -7,18 +7,18 @@ include("../testutils.jl") D = Differential(t) @variables RHS(t) @named fol_separate = ODESystem([RHS ~ (1 - x) / τ, - D(x) ~ RHS]) + D(x) ~ RHS]) fol_simplified = structural_simplify(fol_separate) prob = ODEProblem(fol_simplified, [x => 0.0], (0.0, 10.0), [τ => 3.0]) sol = solve(prob, Tsit5()) sol_new = DiffEqArray(sol.u[1:10], - sol.t[1:10], - sol.prob.f.syms, - sol.prob.f.indepsym, - sol.prob.f.observed, - sol.prob.p) + sol.t[1:10], + sol.prob.f.syms, + sol.prob.f.indepsym, + sol.prob.f.observed, + sol.prob.p) @test sol_new[RHS] ≈ (1 .- sol_new[x]) ./ 3.0 @test sol_new[t] ≈ sol_new.t @@ -31,14 +31,14 @@ test_tables_interface(sol_new, [:timestamp, Symbol("x(t)")], hcat(sol_new[t], so @variables y(t) @parameters α β γ δ @named lv = ODESystem([D(x) ~ α * x - β * x * y, - D(y) ~ δ * x * y - γ * x * y]) + D(y) ~ δ * x * y - γ * x * y]) prob = ODEProblem(lv, [x => 1.0, y => 1.0], (0.0, 10.0), - [α => 1.5, β => 1.0, γ => 3.0, δ => 1.0]) + [α => 1.5, β => 1.0, γ => 3.0, δ => 1.0]) sol = solve(prob, Tsit5()) ts = 0:0.5:10 sol_ts = sol(ts) @assert sol_ts isa DiffEqArray test_tables_interface(sol_ts, [:timestamp, Symbol("x(t)"), Symbol("y(t)")], - hcat(ts, Array(sol_ts)')) + hcat(ts, Array(sol_ts)')) diff --git a/test/partitions_and_static_arrays.jl b/test/partitions_and_static_arrays.jl index e26788b4..d5106161 100644 --- a/test/partitions_and_static_arrays.jl +++ b/test/partitions_and_static_arrays.jl @@ -1,7 +1,7 @@ using Test, RecursiveArrayTools, StaticArrays p = ArrayPartition((zeros(Float32, 2), zeros(SMatrix{2, 2, Int64}, 2), - zeros(SVector{3, Float64}, 2))) + zeros(SVector{3, Float64}, 2))) @test eltype(p) == Float64 @test recursive_bottom_eltype(p) == Float64 @test recursive_unitless_eltype(p) == Float64 diff --git a/test/partitions_test.jl b/test/partitions_test.jl index cb706638..87bd334a 100644 --- a/test/partitions_test.jl +++ b/test/partitions_test.jl @@ -165,7 +165,7 @@ Base.IndexStyle(::MyType) = IndexLinear() Base.BroadcastStyle(::Type{<:MyType}) = Broadcast.ArrayStyle{MyType}() function Base.similar(bc::Broadcast.Broadcasted{Broadcast.ArrayStyle{MyType}}, - ::Type{T}) where {T} + ::Type{T}) where {T} similar(find_mt(bc), T) end @@ -196,20 +196,20 @@ up = 2 .* ap .+ 1 @test typeof(ap) == typeof(up) @testset "ArrayInterface.ismutable(ArrayPartition($a, $b)) == $r" for (a, b, r) in ((1, - 2, - false), - ([ - 1, - ], - 2, - false), - ([ - 1, - ], - [ - 2, - ], - true)) + 2, + false), + ([ + 1, + ], + 2, + false), + ([ + 1, + ], + [ + 2, + ], + true)) @test ArrayInterface.ismutable(ArrayPartition(a, b)) == r end @@ -231,12 +231,12 @@ begin @test_throws MethodError convert(new_type, ArrayPartition(view(b, :), c, c)) end - @testset "Copy and zero with type changing array" begin # Motivating use case for this is ArrayPartitions of Arrow arrays which are mmap:ed and change type when copied struct TypeChangingArray{T, N} <: AbstractArray{T, N} end - Base.copy(::TypeChangingArray{T, N}) where {T,N} = Array{T,N}(undef, ntuple(_ -> 0, N)) - Base.zero(::TypeChangingArray{T, N}) where {T,N} = zeros(T, ntuple(_ -> 0, N)) + Base.copy(::TypeChangingArray{T, N}) where {T, N} = Array{T, N}(undef, + ntuple(_ -> 0, N)) + Base.zero(::TypeChangingArray{T, N}) where {T, N} = zeros(T, ntuple(_ -> 0, N)) a = ArrayPartition(TypeChangingArray{Int, 2}(), TypeChangingArray{Float32, 2}()) @test copy(a) == ArrayPartition(zeros(Int, 0, 0), zeros(Float32, 0, 0)) @@ -245,5 +245,5 @@ end @test !iszero(ArrayPartition([2], [3, 4])) @testset "Cartesian indexing" begin - @test ArrayPartition([1,2], [3])[1:3,1] == [1, 2, 3] + @test ArrayPartition([1, 2], [3])[1:3, 1] == [1, 2, 3] end diff --git a/test/runtests.jl b/test/runtests.jl index 5a45db42..f725cc30 100644 --- a/test/runtests.jl +++ b/test/runtests.jl @@ -26,29 +26,59 @@ end @time begin if GROUP == "Core" || GROUP == "All" - @time @safetestset "Utils Tests" begin include("utils_test.jl") end - @time @safetestset "Partitions Tests" begin include("partitions_test.jl") end - @time @safetestset "VecOfArr Indexing Tests" begin include("basic_indexing.jl") end - @time @safetestset "SymbolicIndexingInterface API test" begin include("symbolic_indexing_interface_test.jl") end - @time @safetestset "VecOfArr Interface Tests" begin include("interface_tests.jl") end - @time @safetestset "Table traits" begin include("tabletraits.jl") end - @time @safetestset "StaticArrays Tests" begin include("copy_static_array_test.jl") end - @time @safetestset "Linear Algebra Tests" begin include("linalg.jl") end - @time @safetestset "Upstream Tests" begin include("upstream.jl") end - @time @safetestset "Adjoint Tests" begin include("adjoints.jl") end - @time @safetestset "Measurement Tests" begin include("measurements.jl") end + @time @safetestset "Utils Tests" begin + include("utils_test.jl") + end + @time @safetestset "Partitions Tests" begin + include("partitions_test.jl") + end + @time @safetestset "VecOfArr Indexing Tests" begin + include("basic_indexing.jl") + end + @time @safetestset "SymbolicIndexingInterface API test" begin + include("symbolic_indexing_interface_test.jl") + end + @time @safetestset "VecOfArr Interface Tests" begin + include("interface_tests.jl") + end + @time @safetestset "Table traits" begin + include("tabletraits.jl") + end + @time @safetestset "StaticArrays Tests" begin + include("copy_static_array_test.jl") + end + @time @safetestset "Linear Algebra Tests" begin + include("linalg.jl") + end + @time @safetestset "Upstream Tests" begin + include("upstream.jl") + end + # @time @safetestset "Adjoint Tests" begin include("adjoints.jl") end + @time @safetestset "Measurement Tests" begin + include("measurements.jl") + end end if !is_APPVEYOR && GROUP == "Downstream" activate_downstream_env() - @time @safetestset "DiffEqArray Indexing Tests" begin include("downstream/symbol_indexing.jl") end - @time @safetestset "Event Tests with ArrayPartition" begin include("downstream/downstream_events.jl") end - VERSION >= v"1.9" && @time @safetestset "Measurements and Units" begin include("downstream/measurements_and_units.jl") end - @time @safetestset "TrackerExt" begin include("downstream/TrackerExt.jl") end + @time @safetestset "DiffEqArray Indexing Tests" begin + include("downstream/symbol_indexing.jl") + end + @time @safetestset "Event Tests with ArrayPartition" begin + include("downstream/downstream_events.jl") + end + VERSION >= v"1.9" && @time @safetestset "Measurements and Units" begin + include("downstream/measurements_and_units.jl") + end + @time @safetestset "TrackerExt" begin + include("downstream/TrackerExt.jl") + end end if !is_APPVEYOR && GROUP == "GPU" activate_gpu_env() - @time @safetestset "VectorOfArray GPU" begin include("gpu/vectorofarray_gpu.jl") end + @time @safetestset "VectorOfArray GPU" begin + include("gpu/vectorofarray_gpu.jl") + end end end diff --git a/test/symbolic_indexing_interface_test.jl b/test/symbolic_indexing_interface_test.jl index 50b5d96d..4b5a2ede 100644 --- a/test/symbolic_indexing_interface_test.jl +++ b/test/symbolic_indexing_interface_test.jl @@ -4,12 +4,18 @@ t = 0.0:0.1:1.0 f(x) = 2x f2(x) = 3x -dx = DiffEqArray([[f(x), f2(x)] for x in t], t; variables = [:a, :b], independent_variables = [:t]) +dx = DiffEqArray([[f(x), f2(x)] for x in t], + t; + variables = [:a, :b], + independent_variables = [:t]) @test dx[:t] == t @test dx[:a] == [f(x) for x in t] @test dx[:b] == [f2(x) for x in t] -dx = DiffEqArray([[f(x), f2(x)] for x in t], t; variables = [:a, :b], independent_variables = [:t]) +dx = DiffEqArray([[f(x), f2(x)] for x in t], + t; + variables = [:a, :b], + independent_variables = [:t]) @test dx[:t] == t dx = DiffEqArray([[f(x), f2(x)] for x in t], t; variables = [:a, :b]) @test_throws Exception dx[nothing] # make sure it isn't storing [nothing] as indepsym diff --git a/test/testutils.jl b/test/testutils.jl index 471b65e1..6917b9c7 100644 --- a/test/testutils.jl +++ b/test/testutils.jl @@ -4,7 +4,7 @@ using RecursiveArrayTools: Tables, IteratorInterfaceExtensions # Test Tables interface with row access + IteratorInterfaceExtensions for QueryVerse # (see https://tables.juliadata.org/stable/#Testing-Tables.jl-Implementations) function test_tables_interface(x::AbstractDiffEqArray, names::Vector{Symbol}, - values::Matrix) + values::Matrix) @assert length(names) == size(values, 2) # AbstractDiffEqArray is a table with row access diff --git a/test/upstream.jl b/test/upstream.jl index 467fa8ff..ad79fd94 100644 --- a/test/upstream.jl +++ b/test/upstream.jl @@ -33,22 +33,22 @@ nlsolve(mymodel, u0) function dyn(u, p, t) ArrayPartition(ArrayPartition(zeros(1), [0.0]), - ArrayPartition(zeros(1), [0.0])) + ArrayPartition(zeros(1), [0.0])) end @test solve(ODEProblem(dyn, - ArrayPartition(ArrayPartition(zeros(1), [-1.0]), - ArrayPartition(zeros(1), [0.75])), - (0.0, 1.0)), AutoTsit5(Rodas5())).retcode == ReturnCode.Success + ArrayPartition(ArrayPartition(zeros(1), [-1.0]), + ArrayPartition(zeros(1), [0.75])), + (0.0, 1.0)), AutoTsit5(Rodas5())).retcode == ReturnCode.Success if VERSION < v"1.7" @test solve(ODEProblem(dyn, - ArrayPartition(ArrayPartition(zeros(1), [-1.0]), - ArrayPartition(zeros(1), [0.75])), - (0.0, 1.0)), Rodas5()).retcode == ReturnCode.Success + ArrayPartition(ArrayPartition(zeros(1), [-1.0]), + ArrayPartition(zeros(1), [0.75])), + (0.0, 1.0)), Rodas5()).retcode == ReturnCode.Success else @test_broken solve(ODEProblem(dyn, - ArrayPartition(ArrayPartition(zeros(1), [-1.0]), - ArrayPartition(zeros(1), [0.75])), - (0.0, 1.0)), Rodas5()).retcode == ReturnCode.Success + ArrayPartition(ArrayPartition(zeros(1), [-1.0]), + ArrayPartition(zeros(1), [0.75])), + (0.0, 1.0)), Rodas5()).retcode == ReturnCode.Success end diff --git a/test/utils_test.jl b/test/utils_test.jl index 01bad869..1d25ecbc 100644 --- a/test/utils_test.jl +++ b/test/utils_test.jl @@ -9,7 +9,7 @@ data = convert(Array, randomized) ## Test means A = [[1 2; 3 4], [1 3; 4 6], [5 6; 7 8]] @test recursive_mean(A) ≈ [2.33333333 3.666666666 - 4.6666666666 6.0] + 4.6666666666 6.0] A = zeros(5, 5) @test recursive_unitless_eltype(A) == Float64 From b8ebfabae65668d971225917616a0a37425df1a4 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Thu, 9 Nov 2023 14:59:54 +0530 Subject: [PATCH 25/45] feat: deprecate linear indexing, fix tests - linear indexing is deprecated - firstindex and lastindex methods removed - eachindex iterates over `CartesianIndex` pointing to individual elements --- src/vector_of_array.jl | 108 ++++++++++++++++++++-------------------- test/basic_indexing.jl | 35 +++++++------ test/interface_tests.jl | 12 ++--- test/utils_test.jl | 4 +- 4 files changed, 82 insertions(+), 77 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index f6542e64..59b7ebfd 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -246,40 +246,25 @@ function SymbolicIndexingInterface.constant_structure(A::DiffEqArray) return constant_structure(A.sys) end -# Interface for the linear indexing. This is just a view of the underlying nested structure -@inline Base.firstindex(VA::AbstractVectorOfArray) = firstindex(VA.u) -@inline Base.lastindex(VA::AbstractVectorOfArray) = lastindex(VA.u) - -@inline Base.length(VA::AbstractVectorOfArray) = length(VA.u) -@inline Base.eachindex(VA::AbstractVectorOfArray) = Base.OneTo(length(VA.u)) -@inline Base.IteratorSize(VA::AbstractVectorOfArray) = Base.HasLength() -# Linear indexing will be over the container elements, not the individual elements -# unlike an true AbstractArray -Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, - I::Int) where {T, N} - VA.u[I] -end -Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, - I::Colon) where {T, N} - VA.u[I] -end -Base.@propagate_inbounds function Base.getindex(VA::AbstractDiffEqArray{T, N}, - I::Colon) where {T, N} - VA.u[I] -end -Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, - I::AbstractArray{Int}) where {T, N} - VectorOfArray(VA.u[I]) -end -Base.@propagate_inbounds function Base.getindex(VA::AbstractDiffEqArray{T, N}, - I::AbstractArray{Int}) where {T, N} - DiffEqArray(VA.u[I], VA.t[I]) +Base.IndexStyle(::Type{<:AbstractVectorOfArray}) = IndexCartesian() + +@inline Base.length(VA::AbstractVectorOfArray) = prod(length.(VA.u)) +@inline function Base.eachindex(VA::AbstractVectorOfArray) + return Iterators.flatten((CartesianIndex(i, j) for i in eachindex(arr)) for (j, arr) in enumerate(VA.u)) end +@inline Base.IteratorSize(::Type{<:AbstractVectorOfArray}) = Base.HasLength() + +@deprecate Base.getindex(A::AbstractVectorOfArray, I::Int) Base.getindex(A, :, I) false +@deprecate Base.getindex(A::AbstractVectorOfArray, I::AbstractArray{Int}) Base.getindex(A, :, I) false +@deprecate Base.getindex(A::AbstractDiffEqArray, I::AbstractArray{Int}) Base.getindex(A, :, I) false Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, I::Union{Int, AbstractArray{Int}, CartesianIndex, Colon, BitArray, AbstractArray{Bool}}...) where {T, N} + if length(I) == 1 + Base.depwarn("Linear indexing of `AbstractDiffEqArray` is deprecated", :getindex) + end RecursiveArrayTools.VectorOfArray(A.u)[I...] end @@ -308,9 +293,17 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray, args.. end end +Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray, ::Colon, I::AbstractArray{Int}) + VectorOfArray(A.u[I]) +end + +Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, ::Colon, I::AbstractArray{Int}) where {T, N} + DiffEqArray(A.u[I], A.t[I], A.p, A.sys) +end + Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, i::Int, ::Colon) where {T, N} - [A.u[j][i] for j in 1:length(A)] + [A.u[j][i] for j in 1:length(A.u)] end Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, ::Colon, i::Int) where {T, N} @@ -367,10 +360,9 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, I::Int...) where {T, N} A.u[I[end]][Base.front(I)...] end -Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, - i::Int) where {T, N} - A.u[i] -end + +@deprecate Base.getindex(A::AbstractDiffEqArray, i::Int) Base.getindex(A, :, i) false + Base.@propagate_inbounds function Base.getindex(VA::AbstractDiffEqArray{T, N}, ii::CartesianIndex) where {T, N} ti = Tuple(ii) @@ -391,7 +383,7 @@ end Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, i::Int, ::Colon) where {T, N} - [VA.u[j][i] for j in 1:length(VA)] + [VA.u[j][i] for j in 1:length(VA.u)] end Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, ii::CartesianIndex) where {T, N} @@ -401,20 +393,29 @@ Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, return VA.u[i][jj] end Base.@propagate_inbounds function Base.setindex!(VA::AbstractVectorOfArray{T, N}, v, - I::Int) where {T, N} + ::Colon, I::Int) where {T, N} VA.u[I] = v end + +@deprecate Base.setindex!(VA::AbstractVectorOfArray, v, I::Int) Base.setindex!(VA, v, :, I) false + Base.@propagate_inbounds function Base.setindex!(VA::AbstractVectorOfArray{T, N}, v, - I::Colon) where {T, N} + ::Colon, I::Colon) where {T, N} VA.u[I] = v end + +@deprecate Base.setindex!(VA::AbstractVectorOfArray, v, I::Colon) Base.setindex!(VA, v, :, I) false + Base.@propagate_inbounds function Base.setindex!(VA::AbstractVectorOfArray{T, N}, v, - I::AbstractArray{Int}) where {T, N} + ::Colon, I::AbstractArray{Int}) where {T, N} VA.u[I] = v end + +@deprecate Base.setindex!(VA::AbstractVectorOfArray, v, I::AbstractArray{Int}) Base.setindex!(VA, v, :, I) false + Base.@propagate_inbounds function Base.setindex!(VA::AbstractVectorOfArray{T, N}, v, i::Int, ::Colon) where {T, N} - for j in 1:length(VA) + for j in 1:length(VA.u) VA.u[j][i] = v[j] end return v @@ -431,6 +432,7 @@ end @inline Base.size(VA::AbstractVectorOfArray) = (size(VA.u[1])..., length(VA.u)) Base.axes(VA::AbstractVectorOfArray) = Base.OneTo.(size(VA)) Base.axes(VA::AbstractVectorOfArray, d::Int) = Base.OneTo(size(VA)[d]) + Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, I::Int...) where {T, N} VA.u[I[end]][Base.front(I)...] @@ -455,7 +457,7 @@ Base.:(==)(A::AbstractArray, B::AbstractVectorOfArray) = B == A # The iterator will be over the subarrays of the container, not the individual elements # unlike an true AbstractArray function Base.iterate(VA::AbstractVectorOfArray, state = 1) - state >= length(VA.u) + 1 ? nothing : (VA[state], state + 1) + state >= length(VA.u) + 1 ? nothing : (VA[:, state], state + 1) end tuples(VA::DiffEqArray) = tuple.(VA.t, VA.u) @@ -543,18 +545,18 @@ end @inline function Base.similar(VA::VectorOfArray, ::Type{T} = eltype(VA), dims = size(VA)) where {T} - VectorOfArray([similar(VA[i], T, Base.front(dims)) for i in 1:last(dims)]) + VectorOfArray([similar(VA[:, i], T, Base.front(dims)) for i in 1:last(dims)]) end recursivecopy(VA::VectorOfArray) = VectorOfArray(copy.(VA.u)) # fill! # For DiffEqArray it ignores ts and fills only u function Base.fill!(VA::AbstractVectorOfArray, x) - for i in eachindex(VA) - if VA[i] isa AbstractArray - fill!(VA[i], x) + for i in 1:length(VA.u) + if VA[:, i] isa AbstractArray + fill!(VA[:, i], x) else - VA[i] = x + VA[:, i] = x end end return VA @@ -567,13 +569,13 @@ function Base._reshape(parent::VectorOfArray, dims::Base.Dims) end # Need this for ODE_DEFAULT_UNSTABLE_CHECK from DiffEqBase to work properly -@inline Base.any(f, VA::AbstractVectorOfArray) = any(any(f, VA[i]) for i in eachindex(VA)) -@inline Base.all(f, VA::AbstractVectorOfArray) = all(all(f, VA[i]) for i in eachindex(VA)) +@inline Base.any(f, VA::AbstractVectorOfArray) = any(f, VA[i] for i in eachindex(VA)) +@inline Base.all(f, VA::AbstractVectorOfArray) = all(f, VA[i] for i in eachindex(VA)) @inline function Base.any(f::Function, VA::AbstractVectorOfArray) - any(any(f, VA[i]) for i in eachindex(VA)) + any(f, VA[i] for i in eachindex(VA)) end @inline function Base.all(f::Function, VA::AbstractVectorOfArray) - all(all(f, VA[i]) for i in eachindex(VA)) + all(f, VA[i] for i in eachindex(VA)) end # conversion tools @@ -672,10 +674,10 @@ end bc = Broadcast.flatten(bc) N = narrays(bc) @inbounds for i in 1:N - if dest[i] isa AbstractArray - copyto!(dest[i], unpack_voa(bc, i)) + if dest[:, i] isa AbstractArray + copyto!(dest[:, i], unpack_voa(bc, i)) else - dest[i] = copy(unpack_voa(bc, i)) + dest[:, i] = copy(unpack_voa(bc, i)) end end dest @@ -689,7 +691,7 @@ end Retrieve number of arrays in the AbstractVectorOfArrays of a broadcast. """ narrays(A) = 0 -narrays(A::AbstractVectorOfArray) = length(A) +narrays(A::AbstractVectorOfArray) = length(A.u) narrays(bc::Broadcast.Broadcasted) = _narrays(bc.args) narrays(A, Bs...) = common_length(narrays(A), _narrays(Bs)) @@ -700,7 +702,7 @@ function common_length(a, b) throw(DimensionMismatch("number of arrays must be equal")))) end -_narrays(args::AbstractVectorOfArray) = length(args) +_narrays(args::AbstractVectorOfArray) = length(args.u) @inline _narrays(args::Tuple) = common_length(narrays(args[1]), _narrays(Base.tail(args))) _narrays(args::Tuple{Any}) = _narrays(args[1]) _narrays(::Any) = 0 diff --git a/test/basic_indexing.jl b/test/basic_indexing.jl index c69d6a85..fe326983 100644 --- a/test/basic_indexing.jl +++ b/test/basic_indexing.jl @@ -37,16 +37,19 @@ testa = cat(recs..., dims = 2) testva = VectorOfArray(recs) # ## Linear indexing -@test testva[1] == testa[:, 1] -@test testva[:] == recs -@test testva[end] == testa[:, end] -@test testva[2:end] == VectorOfArray([recs[i] for i in 2:length(recs)]) +@test_deprecated testva[1] +@test_deprecated testva[1:2] +@test testa[:, 1] == recs[1] +@test testva.u == recs +@test testva[: ,2:end] == VectorOfArray([recs[i] for i in 2:length(recs)]) diffeq = DiffEqArray(recs, t) -@test diffeq[1] == testa[:, 1] -@test diffeq[:] == recs -@test diffeq[end] == testa[:, end] -@test diffeq[2:end] == DiffEqArray([recs[i] for i in 2:length(recs)], t) +@test_deprecated diffeq[1] +@test_deprecated diffeq[1:2] +@test diffeq[:, 1] == testa[:, 1] +@test diffeq.u == recs +@test diffeq[:, end] == testa[:, end] +@test diffeq[:, 2:end] == DiffEqArray([recs[i] for i in 2:length(recs)], t) # ## (Int, Int) @test testa[5, 4] == testva[5, 4] @@ -139,10 +142,10 @@ v[CartesianIndex((2, 3, 2, 3))] = 1 v = VectorOfArray([rand(20), rand(10, 10), rand(3, 3, 3)]) w = v .* v @test w isa VectorOfArray -@test w[1] isa Vector -@test w[1] == v[1] .* v[1] -@test w[2] == v[2] .* v[2] -@test w[3] == v[3] .* v[3] +@test w[:, 1] isa Vector +@test w[:, 1] == v[:, 1] .* v[:, 1] +@test w[:, 2] == v[:, 2] .* v[:, 2] +@test w[:, 3] == v[:, 3] .* v[:, 3] x = copy(v) x .= v .* v @test x.u == w.u @@ -153,10 +156,10 @@ w = v .+ 1 v = DiffEqArray([rand(20), rand(10, 10), rand(3, 3, 3)], 1:3) w = v .* v @test_broken w isa DiffEqArray # FIXME -@test w[1] isa Vector -@test w[1] == v[1] .* v[1] -@test w[2] == v[2] .* v[2] -@test w[3] == v[3] .* v[3] +@test w[:, 1] isa Vector +@test w[:, 1] == v[:, 1] .* v[:, 1] +@test w[:, 2] == v[:, 2] .* v[:, 2] +@test w[:, 3] == v[:, 3] .* v[:, 3] x = copy(v) x .= v .* v @test x.u == w.u diff --git a/test/interface_tests.jl b/test/interface_tests.jl index 3ac5283d..f87bb5dd 100644 --- a/test/interface_tests.jl +++ b/test/interface_tests.jl @@ -5,11 +5,11 @@ testva = VectorOfArray([[1, 2, 3], [4, 5, 6], [7, 8, 9]]) testda = DiffEqArray([[1, 2, 3], [4, 5, 6], [7, 8, 9]], t) for (i, elem) in enumerate(testva) - @test elem == testva[i] + @test elem == testva[:, i] end for (i, elem) in enumerate(testda) - @test elem == testda[i] + @test elem == testda[:, i] end push!(testva, [10, 11, 12]) @@ -43,10 +43,10 @@ push!(testda, [-1, -2, -3, -4]) @test_throws BoundsError testva[4:5, 5:6] @test_throws BoundsError testda[4:5, 5:6] -@test testva[9] == [-1, -2, -3, -4] -@test testva[end] == [-1, -2, -3, -4] -@test testda[9] == [-1, -2, -3, -4] -@test testda[end] == [-1, -2, -3, -4] +@test testva[:, 9] == [-1, -2, -3, -4] +@test testva[:, end] == [-1, -2, -3, -4] +@test testda[:, 9] == [-1, -2, -3, -4] +@test testda[:, end] == [-1, -2, -3, -4] # Currently we enforce the general shape, they can just be different lengths, ie we # can't do diff --git a/test/utils_test.jl b/test/utils_test.jl index 1d25ecbc..9769c523 100644 --- a/test/utils_test.jl +++ b/test/utils_test.jl @@ -107,9 +107,9 @@ recursivefill!(x, true) y_voa = recursivecopy(x_voa) recursivefill!(y_voa, true) - @test all(y_voa[n] == fill(ones(Vec3), n) for n in 1:4) + @test all(y_voa[:, n] == fill(ones(Vec3), n) for n in 1:4) y_voa = recursivecopy(x_voa) recursivefill!(y_voa, ones(Vec3)) - @test all(y_voa[n] == fill(ones(Vec3), n) for n in 1:4) + @test all(y_voa[:, n] == fill(ones(Vec3), n) for n in 1:4) end From 3cb88edad2b33e63f28f804147c7f7ededa57390 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Thu, 9 Nov 2023 20:13:14 +0530 Subject: [PATCH 26/45] refactor: drastically simplify getindex methods --- src/vector_of_array.jl | 89 +++++++++++++----------------------------- 1 file changed, 27 insertions(+), 62 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 59b7ebfd..81c7e503 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -254,19 +254,14 @@ Base.IndexStyle(::Type{<:AbstractVectorOfArray}) = IndexCartesian() end @inline Base.IteratorSize(::Type{<:AbstractVectorOfArray}) = Base.HasLength() + @deprecate Base.getindex(A::AbstractVectorOfArray, I::Int) Base.getindex(A, :, I) false + @deprecate Base.getindex(A::AbstractVectorOfArray, I::AbstractArray{Int}) Base.getindex(A, :, I) false + @deprecate Base.getindex(A::AbstractDiffEqArray, I::AbstractArray{Int}) Base.getindex(A, :, I) false -Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, - I::Union{Int, AbstractArray{Int}, - CartesianIndex, Colon, BitArray, - AbstractArray{Bool}}...) where {T, - N} - if length(I) == 1 - Base.depwarn("Linear indexing of `AbstractDiffEqArray` is deprecated", :getindex) - end - RecursiveArrayTools.VectorOfArray(A.u)[I...] -end + +@deprecate Base.getindex(A::AbstractDiffEqArray, i::Int) Base.getindex(A, :, i) false __parameterless_type(T) = Base.typename(T).wrapper Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray{T, N}, @@ -285,34 +280,36 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray{T, N}, return A[ntuple(x -> Colon(), ndims(A))...][I, J...] end -Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray, args...) - if last(args) isa Union{Integer, CartesianIndex} - return getindex(A.u[last(args)], Base.front(args)...) - else - return stack(getindex.(A.u[last(args)], tuple.(Base.front(args))...)) +# Need two of each methods to avoid ambiguities +for voa in [AbstractVectorOfArray, AbstractDiffEqArray] + @eval Base.@propagate_inbounds function Base.getindex(A::$(voa), ::Colon, I::Int) + A.u[I] end + + @eval Base.@propagate_inbounds function Base.getindex(A::$(voa), I::Union{Int,AbstractArray{Int},AbstractArray{Bool},Colon}...) + if last(I) isa Int + A.u[last(I)][Base.front(I)...] + else + stack(getindex.(A.u[last(I)], tuple.(Base.front(I))...)) + end + end + @eval Base.@propagate_inbounds function Base.getindex(VA::$(voa), ii::CartesianIndex) + ti = Tuple(ii) + i = last(ti) + jj = CartesianIndex(Base.front(ti)) + return VA.u[i][jj] + end + end -Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray, ::Colon, I::AbstractArray{Int}) +Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray, ::Colon, I::Union{AbstractArray{Int},AbstractArray{Bool}}) VectorOfArray(A.u[I]) end -Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, ::Colon, I::AbstractArray{Int}) where {T, N} +Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray, ::Colon, I::Union{AbstractArray{Int},AbstractArray{Bool}}) DiffEqArray(A.u[I], A.t[I], A.p, A.sys) end -Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, i::Int, - ::Colon) where {T, N} - [A.u[j][i] for j in 1:length(A.u)] -end -Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, ::Colon, - i::Int) where {T, N} - A.u[i] -end -Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, i::Int, - II::AbstractArray{Int}) where {T, N} - [A.u[j][i] for j in II] -end Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, sym) where {T, N} if is_independent_variable(A, sym) @@ -338,6 +335,7 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, end return getindex.(A.u, sym) end + Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, sym, args...) where {T, N} A.sys === nothing && error("Cannot use symbolic indexing without a system") @@ -356,20 +354,6 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, sy return reduce(vcat, map(s -> A[s, args...]', sym)) end end -Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, - I::Int...) where {T, N} - A.u[I[end]][Base.front(I)...] -end - -@deprecate Base.getindex(A::AbstractDiffEqArray, i::Int) Base.getindex(A, :, i) false - -Base.@propagate_inbounds function Base.getindex(VA::AbstractDiffEqArray{T, N}, - ii::CartesianIndex) where {T, N} - ti = Tuple(ii) - i = last(ti) - jj = CartesianIndex(Base.front(ti)) - return VA.u[i][jj] -end function _observed(A::AbstractDiffEqArray{T, N}, sym, i::Int) where {T, N} observed(A, sym)(A.u[i], A.p, A.t[i]) @@ -381,17 +365,6 @@ function _observed(A::AbstractDiffEqArray{T, N}, sym, ::Colon) where {T, N} observed(A, sym).(A.u, (A.p,), A.t) end -Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, i::Int, - ::Colon) where {T, N} - [VA.u[j][i] for j in 1:length(VA.u)] -end -Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, - ii::CartesianIndex) where {T, N} - ti = Tuple(ii) - i = last(ti) - jj = CartesianIndex(Base.front(ti)) - return VA.u[i][jj] -end Base.@propagate_inbounds function Base.setindex!(VA::AbstractVectorOfArray{T, N}, v, ::Colon, I::Int) where {T, N} VA.u[I] = v @@ -433,14 +406,6 @@ end Base.axes(VA::AbstractVectorOfArray) = Base.OneTo.(size(VA)) Base.axes(VA::AbstractVectorOfArray, d::Int) = Base.OneTo(size(VA)[d]) -Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, - I::Int...) where {T, N} - VA.u[I[end]][Base.front(I)...] -end -Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray{T, N}, ::Colon, - I::Int) where {T, N} - VA.u[I] -end Base.@propagate_inbounds function Base.setindex!(VA::AbstractVectorOfArray{T, N}, v, I::Int...) where {T, N} VA.u[I[end]][Base.front(I)...] = v From 975ebfca543128886f6990bc673dea24ec4fe0cc Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Fri, 10 Nov 2023 17:23:31 +0530 Subject: [PATCH 27/45] fix: fix constructor --- src/vector_of_array.jl | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 81c7e503..4e03d8b7 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -182,9 +182,9 @@ function DiffEqArray(vec::AbstractVector{VT}, variables = nothing, parameters = nothing, independent_variables = nothing) where {T, N, VT <: AbstractArray{T, N}} - sys = SymbolCache(something(variables, []), + sys = something(sys, SymbolCache(something(variables, []), something(parameters, []), - something(independent_variables, [])) + something(independent_variables, []))) return DiffEqArray{ eltype(eltype(vec)), N + 1, From 06ba92966ddc7bc47ca5b939f6f249a8fb4b8214 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Fri, 10 Nov 2023 17:23:57 +0530 Subject: [PATCH 28/45] feat: revert length method, add Base.first and Base.last methods --- src/vector_of_array.jl | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 4e03d8b7..844700aa 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -248,12 +248,13 @@ end Base.IndexStyle(::Type{<:AbstractVectorOfArray}) = IndexCartesian() -@inline Base.length(VA::AbstractVectorOfArray) = prod(length.(VA.u)) +@inline Base.length(VA::AbstractVectorOfArray) = length(VA.u) @inline function Base.eachindex(VA::AbstractVectorOfArray) return Iterators.flatten((CartesianIndex(i, j) for i in eachindex(arr)) for (j, arr) in enumerate(VA.u)) end @inline Base.IteratorSize(::Type{<:AbstractVectorOfArray}) = Base.HasLength() - +@inline Base.first(VA::AbstractVectorOfArray) = first(VA.u) +@inline Base.last(VA::AbstractVectorOfArray) = last(VA.u) @deprecate Base.getindex(A::AbstractVectorOfArray, I::Int) Base.getindex(A, :, I) false From 6e1512287f19ebbfdcac0405ecdd15c364eaee46 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Fri, 10 Nov 2023 17:25:02 +0530 Subject: [PATCH 29/45] feat: use trait dispatch for getindex method - resolves method ambiguities - reduces total number of methods - also added a minimal interface for DiffEqArray --- src/vector_of_array.jl | 120 +++++++++++++++++++++++++---------------- 1 file changed, 74 insertions(+), 46 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 844700aa..3dbeffca 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -198,6 +198,10 @@ function DiffEqArray(vec::AbstractVector{VT}, sys) end +# AbstractDiffEqArray Interface +parameter_values(A::AbstractDiffEqArray) = A.p +symbolic_container(A::AbstractDiffEqArray) = A.sys + # SymbolicIndexingInterface implementation for DiffEqArray # Just forward to A.sys function SymbolicIndexingInterface.is_variable(A::DiffEqArray, sym) @@ -266,7 +270,7 @@ end __parameterless_type(T) = Base.typename(T).wrapper Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray{T, N}, - I::Colon...) where {T, N} + ::NotSymbolic, I::Colon...) where {T, N} @assert length(I) == ndims(A.u[1]) + 1 vecs = vec.(A.u) return Adapt.adapt(__parameterless_type(T), @@ -274,7 +278,7 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray{T, N}, end Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray{T, N}, - I::AbstractArray{Bool}, + ::NotSymbolic, I::AbstractArray{Bool}, J::Colon...) where {T, N} @assert length(J) == ndims(A.u[1]) + 1 - ndims(I) @assert size(I) == size(A)[1:(ndims(A) - length(J))] @@ -282,37 +286,34 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray{T, N}, end # Need two of each methods to avoid ambiguities -for voa in [AbstractVectorOfArray, AbstractDiffEqArray] - @eval Base.@propagate_inbounds function Base.getindex(A::$(voa), ::Colon, I::Int) - A.u[I] - end +Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray, ::NotSymbolic, ::Colon, I::Int) + A.u[I] +end - @eval Base.@propagate_inbounds function Base.getindex(A::$(voa), I::Union{Int,AbstractArray{Int},AbstractArray{Bool},Colon}...) - if last(I) isa Int - A.u[last(I)][Base.front(I)...] - else - stack(getindex.(A.u[last(I)], tuple.(Base.front(I))...)) - end - end - @eval Base.@propagate_inbounds function Base.getindex(VA::$(voa), ii::CartesianIndex) - ti = Tuple(ii) - i = last(ti) - jj = CartesianIndex(Base.front(ti)) - return VA.u[i][jj] +Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray, ::NotSymbolic, I::Union{Int,AbstractArray{Int},AbstractArray{Bool},Colon}...) + if last(I) isa Int + A.u[last(I)][Base.front(I)...] + else + stack(getindex.(A.u[last(I)], tuple.(Base.front(I))...)) end - +end +Base.@propagate_inbounds function Base.getindex(VA::AbstractVectorOfArray, ::NotSymbolic, ii::CartesianIndex) + ti = Tuple(ii) + i = last(ti) + jj = CartesianIndex(Base.front(ti)) + return VA.u[i][jj] end -Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray, ::Colon, I::Union{AbstractArray{Int},AbstractArray{Bool}}) +Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray, ::NotSymbolic, ::Colon, I::Union{AbstractArray{Int},AbstractArray{Bool}}) VectorOfArray(A.u[I]) end -Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray, ::Colon, I::Union{AbstractArray{Int},AbstractArray{Bool}}) - DiffEqArray(A.u[I], A.t[I], A.p, A.sys) +Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray, ::NotSymbolic, ::Colon, I::Union{AbstractArray{Int},AbstractArray{Bool}}) + DiffEqArray(A.u[I], A.t[I], parameter_values(A), symbolic_container(A)) end -Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, - sym) where {T, N} +# Symbolic Indexing Methods +Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray, ::ScalarSymbolic, sym) if is_independent_variable(A, sym) return A.t elseif is_variable(A, sym) @@ -322,37 +323,64 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, return getindex.(A.u, variable_index.((A,), (sym,), eachindex(A.t))) end elseif is_parameter(A, sym) - return A.p[parameter_index(A, sym)] + return parameter_values(A)[parameter_index(A, sym)] elseif is_observed(A, sym) - return observed(A, sym, :) - elseif symbolic_type(sym) == ArraySymbolic() - return getindex(A, collect(sym)) - elseif sym isa AbstractArray - if all(x -> is_parameter(A, x), collect(sym)) - return getindex.((A,), sym) - else - return [getindex.((A,), sym, i) for i in eachindex(A.t)] - end + return observed(A, sym).(A.u, (parameter_values(A),), A.t) + else + # NOTE: this is basically just for LabelledArrays. It's better if this + # were an error. Should we make an extension for LabelledArrays handling + # this case? + return getindex.(A.u, sym) end - return getindex.(A.u, sym) end -Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray{T, N}, sym, - args...) where {T, N} - A.sys === nothing && error("Cannot use symbolic indexing without a system") - +Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray, ::ScalarSymbolic, sym, args...) if is_independent_variable(A, sym) return A.t[args...] - elseif is_variable(A.sys, sym) - if constant_structure(A) - return A[sym][args...] + elseif is_variable(A, sym) + return A[sym][args...] + elseif is_observed(A, sym) + u = A.u[args...] + t = A.t[args...] + observed_fn = observed(A, sym) + if t isa AbstractArray + return observed_fn.(u, (parameter_values(A),), t) else - return getindex.(A.u, variable_index.((A,), (sym,), A.t[args...])) + return observed_fn(u, parameter_values(A), t) end - elseif is_observed(A, sym) - return observed(A, sym, args...) else - return reduce(vcat, map(s -> A[s, args...]', sym)) + # NOTE: this is basically just for LabelledArrays. It's better if this + # were an error. Should we make an extension for LabelledArrays handling + # this case? + return getindex.(A.u[args...], sym) + end +end + + +Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray, ::ArraySymbolic, sym, args...) + return getindex(A, collect(sym), args...) +end + +Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray, ::ScalarSymbolic, sym::Union{Tuple,AbstractArray}) + if all(x -> is_parameter(A, x), sym) + return getindex.((A,), sym) + else + return [getindex.((A,), sym, i) for i in eachindex(A.t)] + end +end + +Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray, ::ScalarSymbolic, sym::Union{Tuple,AbstractArray}, args...) + return reduce(vcat, map(s -> A[s, args...]', sym)) +end + +Base.@propagate_inbounds function Base.getindex(A::AbstractVectorOfArray, _arg, args...) + symtype = symbolic_type(_arg) + elsymtype = symbolic_type(eltype(_arg)) + + if symtype != NotSymbolic() + return Base.getindex(A, symtype, _arg, args...) + else + return Base.getindex(A, elsymtype, _arg, args...) end end From c875b658da32d134c81a4bbcdb70c15a1bf37c41 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Tue, 14 Nov 2023 15:50:44 +0530 Subject: [PATCH 30/45] fix: add firstindex and lastindex methods with depwarn --- src/vector_of_array.jl | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 3dbeffca..e39991e2 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -259,6 +259,15 @@ end @inline Base.IteratorSize(::Type{<:AbstractVectorOfArray}) = Base.HasLength() @inline Base.first(VA::AbstractVectorOfArray) = first(VA.u) @inline Base.last(VA::AbstractVectorOfArray) = last(VA.u) +function Base.firstindex(VA::AbstractVectorOfArray) + Base.depwarn("Linear indexing of `AbstractVectorOfArray` is deprecated", :firstindex) + return firstindex(VA.u) +end + +function Base.lastindex(VA::AbstractVectorOfArray) + Base.depwarn("Linear indexing of `AbstractVectorOfArray` is deprecated", :lastindex) + return lastindex(VA.u) +end @deprecate Base.getindex(A::AbstractVectorOfArray, I::Int) Base.getindex(A, :, I) false From 364a9058be87b3d6a2039492a4f0f8c7c2af9f71 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Tue, 14 Nov 2023 15:50:56 +0530 Subject: [PATCH 31/45] test: fix DiffEqConstructor --- test/downstream/symbol_indexing.jl | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/test/downstream/symbol_indexing.jl b/test/downstream/symbol_indexing.jl index eb3739da..19ee339d 100644 --- a/test/downstream/symbol_indexing.jl +++ b/test/downstream/symbol_indexing.jl @@ -15,10 +15,8 @@ sol = solve(prob, Tsit5()) sol_new = DiffEqArray(sol.u[1:10], sol.t[1:10], - sol.prob.f.syms, - sol.prob.f.indepsym, - sol.prob.f.observed, - sol.prob.p) + sol.prob.p, + sol) @test sol_new[RHS] ≈ (1 .- sol_new[x]) ./ 3.0 @test sol_new[t] ≈ sol_new.t From 53e9fafc605f3150ee5d5cdc15d3b5a07e035865 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Thu, 16 Nov 2023 13:30:07 +0530 Subject: [PATCH 32/45] fix: add IndexStyle method, fix reshape for VectorOfArray --- src/vector_of_array.jl | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index e39991e2..04d6780d 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -250,6 +250,7 @@ function SymbolicIndexingInterface.constant_structure(A::DiffEqArray) return constant_structure(A.sys) end +Base.IndexStyle(A::AbstractVectorOfArray) = Base.IndexStyle(typeof(A)) Base.IndexStyle(::Type{<:AbstractVectorOfArray}) = IndexCartesian() @inline Base.length(VA::AbstractVectorOfArray) = length(VA.u) @@ -565,11 +566,7 @@ function Base.fill!(VA::AbstractVectorOfArray, x) return VA end -function Base._reshape(parent::VectorOfArray, dims::Base.Dims) - n = prod(size(parent)) - prod(dims) == n || Base._throw_dmrs(n, "size", dims) - Base.__reshape((parent, IndexStyle(parent)), dims) -end +Base.reshape(A::VectorOfArray, dims...) = Base.reshape(Array(A), dims...) # Need this for ODE_DEFAULT_UNSTABLE_CHECK from DiffEqBase to work properly @inline Base.any(f, VA::AbstractVectorOfArray) = any(f, VA[i] for i in eachindex(VA)) From 919df69d6e546d91f7b8157ffcc45d974a2d5d87 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Thu, 16 Nov 2023 18:46:31 +0530 Subject: [PATCH 33/45] test: mark measurements and units test as broken --- test/downstream/measurements_and_units.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/test/downstream/measurements_and_units.jl b/test/downstream/measurements_and_units.jl index ded9184e..c3f772f7 100644 --- a/test/downstream/measurements_and_units.jl +++ b/test/downstream/measurements_and_units.jl @@ -6,4 +6,4 @@ tspan3 = (0.0, 1.0) .* u"s" f3(du, u, p, t) = [0.0u"m/s^2", -g3] u3 = [0.0, 0.0] .* u"m" problem4 = SecondOrderODEProblem(f3, du4, u3, tspan3) -solve(problem4, Tsit5()) +@test_broken solve(problem4, Tsit5()) From 2e607b8419db5cb6c7bf7cbb213317230261a16f Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Mon, 20 Nov 2023 17:56:45 +0530 Subject: [PATCH 34/45] fix: fix Base.similar for VectorOfArray --- src/vector_of_array.jl | 10 ++++------ 1 file changed, 4 insertions(+), 6 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 04d6780d..f1e0bb79 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -543,13 +543,11 @@ end # Tools for creating similar objects Base.eltype(::VectorOfArray{T}) where {T} = T -@inline function Base.similar(VA::VectorOfArray, dims::NTuple) - similar(VA, eltype(VA), dims) +@inline function Base.similar(VA::VectorOfArray, args...) + return Base.similar(ones(eltype(VA)), args...) end -@inline function Base.similar(VA::VectorOfArray, - ::Type{T} = eltype(VA), - dims = size(VA)) where {T} - VectorOfArray([similar(VA[:, i], T, Base.front(dims)) for i in 1:last(dims)]) +@inline function Base.similar(VA::VectorOfArray, ::Type{T} = eltype(VA)) where {T} + VectorOfArray([similar(VA[:, i], T) for i in eachindex(VA.u)]) end recursivecopy(VA::VectorOfArray) = VectorOfArray(copy.(VA.u)) From 6a5e303e60cb0de886a2235d45b41e7017d6a9a6 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Mon, 20 Nov 2023 17:59:25 +0530 Subject: [PATCH 35/45] fix!: Julia 1.10 fix for ldiv --- src/array_partition.jl | 24 ------------------------ 1 file changed, 24 deletions(-) diff --git a/src/array_partition.jl b/src/array_partition.jl index 88d37bd9..2bfc777e 100644 --- a/src/array_partition.jl +++ b/src/array_partition.jl @@ -219,18 +219,6 @@ Base.@propagate_inbounds function Base.getindex(A::ArrayPartition, i::Int) end end -""" - getindex(A::ArrayPartition, i::Int, j...) - -Returns the entry at index `j...` of the `i`th partition of `A`. -""" -Base.@propagate_inbounds function Base.getindex(A::ArrayPartition, i::Int, j...) - @boundscheck 0 < i <= length(A.x) || throw(BoundsError(A.x, i)) - @inbounds b = A.x[i] - @boundscheck checkbounds(b, j...) - @inbounds return b[j...] -end - """ getindex(A::ArrayPartition, i::Colon, j...) @@ -274,18 +262,6 @@ function Base._maybe_reshape(::IndexCartesian, Vector(A) end -""" - setindex!(A::ArrayPartition, v, i::Int, j...) - -Set the entry at index `j...` of the `i`th partition of `A` to `v`. -""" -Base.@propagate_inbounds function Base.setindex!(A::ArrayPartition, v, i::Int, j...) - @boundscheck 0 < i <= length(A.x) || throw(BoundsError(A.x, i)) - @inbounds b = A.x[i] - @boundscheck checkbounds(b, j...) - @inbounds b[j...] = v -end - ## recursive methods function recursivecopy!(A::ArrayPartition, B::ArrayPartition) From 12204882dd486a43c1e78fd13769932961503073 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Mon, 20 Nov 2023 17:59:38 +0530 Subject: [PATCH 36/45] refactor: hacky fix for autodiff --- ext/RecursiveArrayToolsZygoteExt.jl | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/ext/RecursiveArrayToolsZygoteExt.jl b/ext/RecursiveArrayToolsZygoteExt.jl index 9434c96d..5f93bf96 100644 --- a/ext/RecursiveArrayToolsZygoteExt.jl +++ b/ext/RecursiveArrayToolsZygoteExt.jl @@ -94,16 +94,20 @@ end @adjoint function VectorOfArray(u) VectorOfArray(u), y -> begin - (VectorOfArray([y[].u[ntuple(x -> Colon(), ndims(y[].u) - 1)..., i] - for i in 1:size(y[].u)[end]]),) + y isa Ref && (y = VectorOfArray(y[].u)) + (VectorOfArray([y[ntuple(x -> Colon(), ndims(y.u) - 1)..., i] + for i in 1:size(y.u)[end]]),) end end @adjoint function DiffEqArray(u, t) DiffEqArray(u, t), - y -> (DiffEqArray([y[].u[ntuple(x -> Colon(), ndims(y[].u) - 1)..., i] - for i in 1:size(y[].u)[end]], + y -> begin + y isa Ref && (y = VectorOfArray(y[].u)) + (DiffEqArray([y[ntuple(x -> Colon(), ndims(y.u) - 1)..., i] + for i in 1:size(y.u)[end]], t), nothing) + end end @adjoint function literal_getproperty(A::ArrayPartition, ::Val{:x}) From d84322270e8c1f1075f8a64911bc57dd09e66f76 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Fri, 24 Nov 2023 15:33:33 +0530 Subject: [PATCH 37/45] refactor: use symbolic_container from SII for fallback implementation --- src/vector_of_array.jl | 52 ++---------------------------------------- 1 file changed, 2 insertions(+), 50 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index f1e0bb79..11f4221c 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -199,56 +199,8 @@ function DiffEqArray(vec::AbstractVector{VT}, end # AbstractDiffEqArray Interface -parameter_values(A::AbstractDiffEqArray) = A.p -symbolic_container(A::AbstractDiffEqArray) = A.sys - -# SymbolicIndexingInterface implementation for DiffEqArray -# Just forward to A.sys -function SymbolicIndexingInterface.is_variable(A::DiffEqArray, sym) - return is_variable(A.sys, sym) -end -function SymbolicIndexingInterface.variable_index(A::DiffEqArray, sym) - return variable_index(A.sys, sym) -end -function SymbolicIndexingInterface.variable_index(A::DiffEqArray, sym, t) - return variable_index(A.sys, sym, t) -end -function SymbolicIndexingInterface.variable_symbols(A::DiffEqArray) - return variable_symbols(A.sys) -end -function SymbolicIndexingInterface.variable_symbols(A::DiffEqArray, i) - return variable_symbols(A.sys, i) -end -function SymbolicIndexingInterface.is_parameter(A::DiffEqArray, sym) - return is_parameter(A.sys, sym) -end -function SymbolicIndexingInterface.parameter_index(A::DiffEqArray, sym) - return parameter_index(A.sys, sym) -end -function SymbolicIndexingInterface.parameter_symbols(A::DiffEqArray) - return parameter_symbols(A.sys) -end -function SymbolicIndexingInterface.is_independent_variable(A::DiffEqArray, sym) - return is_independent_variable(A.sys, sym) -end -function SymbolicIndexingInterface.independent_variable_symbols(A::DiffEqArray) - return independent_variable_symbols(A.sys) -end -function SymbolicIndexingInterface.is_observed(A::DiffEqArray, sym) - return is_observed(A.sys, sym) -end -function SymbolicIndexingInterface.observed(A::DiffEqArray, sym) - return observed(A.sys, sym) -end -function SymbolicIndexingInterface.observed(A::DiffEqArray, sym, symbolic_states) - return observed(A.sys, sym, symbolic_states) -end -function SymbolicIndexingInterface.is_time_dependent(A::DiffEqArray) - return is_time_dependent(A.sys) -end -function SymbolicIndexingInterface.constant_structure(A::DiffEqArray) - return constant_structure(A.sys) -end +SymbolicIndexingInterface.parameter_values(A::AbstractDiffEqArray) = A.p +SymbolicIndexingInterface.symbolic_container(A::AbstractDiffEqArray) = A.sys Base.IndexStyle(A::AbstractVectorOfArray) = Base.IndexStyle(typeof(A)) Base.IndexStyle(::Type{<:AbstractVectorOfArray}) = IndexCartesian() From 79968901da0d3dc8290727a7b30e3d9efcffbfc4 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Mon, 27 Nov 2023 10:45:28 +0530 Subject: [PATCH 38/45] refactor: deprecate direct parameter indexing instead of erroring --- src/vector_of_array.jl | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 11f4221c..67326f6e 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -198,7 +198,14 @@ function DiffEqArray(vec::AbstractVector{VT}, sys) end -# AbstractDiffEqArray Interface +function Base.getproperty(A::AbstractDiffEqArray, sym::Symbol) + if sym == SymbolicIndexingInterface.PARAMETER_INDEXING_PROXY_PROPERTY_NAME + return ParameterIndexingProxy(A) + else + return getfield(A, sym) + end +end + SymbolicIndexingInterface.parameter_values(A::AbstractDiffEqArray) = A.p SymbolicIndexingInterface.symbolic_container(A::AbstractDiffEqArray) = A.sys @@ -285,7 +292,8 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray, ::Scalar return getindex.(A.u, variable_index.((A,), (sym,), eachindex(A.t))) end elseif is_parameter(A, sym) - return parameter_values(A)[parameter_index(A, sym)] + Base.depwarn("Indexing with parameters is deprecated. Use `$(nameof(typeof(A))).$(SymbolicIndexingInterface.PARAMETER_INDEXING_PROXY_PROPERTY_NAME)[$sym]` for parameter indexing.", :parameter_getindex) + A.ps[sym] elseif is_observed(A, sym) return observed(A, sym).(A.u, (parameter_values(A),), A.t) else From 92fc778fd15eae24b37af7cba8090613daa538c6 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Mon, 27 Nov 2023 12:46:25 +0530 Subject: [PATCH 39/45] refactor: fix VectorOfArray arithmetic --- src/vector_of_array.jl | 14 ++++++++++---- 1 file changed, 10 insertions(+), 4 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 67326f6e..c08eee2d 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -476,12 +476,17 @@ function Base.isapprox(A::AbstractArray, B::AbstractVectorOfArray; kwargs...) end for op in [:(Base.:-), :(Base.:+)] - @eval function ($op)(A::AbstractVectorOfArray, - B::Union{AbstractVectorOfArray, AbstractArray}) + @eval function ($op)(A::AbstractVectorOfArray, B::AbstractVectorOfArray) ($op).(A, B) end - @eval function ($op)(A::AbstractArray, B::AbstractVectorOfArray) - ($op).(A, B) + @eval Base.@propagate_inbounds function ($op)(A::AbstractVectorOfArray, + B::AbstractArray) + @boundscheck length(A) == length(B) + VectorOfArray([($op).(a, b) for (a, b) in zip(A, B)]) + end + @eval Base.@propagate_inbounds function ($op)(A::AbstractArray, B::AbstractVectorOfArray) + @boundscheck length(A) == length(B) + VectorOfArray([($op).(a, b) for (a, b) in zip(A, B)]) end end @@ -503,6 +508,7 @@ end # Tools for creating similar objects Base.eltype(::VectorOfArray{T}) where {T} = T +# TODO: Is there a better way to do this? @inline function Base.similar(VA::VectorOfArray, args...) return Base.similar(ones(eltype(VA)), args...) end From cbf61c6187bd18f600145b430cc418b092a60680 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Mon, 27 Nov 2023 12:46:45 +0530 Subject: [PATCH 40/45] test: more comprehensive tests --- test/basic_indexing.jl | 24 +++++++++++++++ test/downstream/symbol_indexing.jl | 3 ++ test/symbolic_indexing_interface_test.jl | 39 +++++++++++++++++++----- 3 files changed, 58 insertions(+), 8 deletions(-) diff --git a/test/basic_indexing.jl b/test/basic_indexing.jl index fe326983..7a5acbe6 100644 --- a/test/basic_indexing.jl +++ b/test/basic_indexing.jl @@ -36,9 +36,33 @@ recs = [rand(8) for i in 1:10] testa = cat(recs..., dims = 2) testva = VectorOfArray(recs) +# Array functions +@test size(testva) == (8, 10) +@test axes(testva) == Base.OneTo.((8, 10)) +@test ndims(testva) == 2 +@test eltype(testva) == eltype(eltype(recs)) +testvasim = similar(testva) +@test size(testvasim) == size(testva) +@test eltype(testvasim) == eltype(testva) +testvasim = similar(testva, Float32) +@test size(testvasim) == size(testva) +@test eltype(testvasim) == Float32 +testvb = deepcopy(testva) +@test testva == testvb == recs + +# Math operations +@test testva + testvb == testva + recs == 2testva == 2 .* recs +@test testva - testvb == testva - recs == 0 .* recs +@test testva / 2 == recs ./ 2 +@test 2 .\ testva == 2 .\ recs + # ## Linear indexing @test_deprecated testva[1] @test_deprecated testva[1:2] +@test_deprecated testva[begin] +@test_deprecated testva[end] +@test testva[begin] == testva[:, begin] == first(testva) +@test testva[end] == testva[:, end] == last(testva) @test testa[:, 1] == recs[1] @test testva.u == recs @test testva[: ,2:end] == VectorOfArray([recs[i] for i in 2:length(recs)]) diff --git a/test/downstream/symbol_indexing.jl b/test/downstream/symbol_indexing.jl index 19ee339d..3c79cc26 100644 --- a/test/downstream/symbol_indexing.jl +++ b/test/downstream/symbol_indexing.jl @@ -21,6 +21,9 @@ sol_new = DiffEqArray(sol.u[1:10], @test sol_new[RHS] ≈ (1 .- sol_new[x]) ./ 3.0 @test sol_new[t] ≈ sol_new.t @test sol_new[t, 1:5] ≈ sol_new.t[1:5] +@test sol.ps[τ] == sol_new.ps[τ] == 3.0 +@test_deprecated sol[τ] +@test_deprecated sol_new[τ] # Tables interface test_tables_interface(sol_new, [:timestamp, Symbol("x(t)")], hcat(sol_new[t], sol_new[x])) diff --git a/test/symbolic_indexing_interface_test.jl b/test/symbolic_indexing_interface_test.jl index 4b5a2ede..66e2f697 100644 --- a/test/symbolic_indexing_interface_test.jl +++ b/test/symbolic_indexing_interface_test.jl @@ -1,22 +1,44 @@ -using RecursiveArrayTools, Test, LabelledArrays +using RecursiveArrayTools, Test, LabelledArrays, SymbolicIndexingInterface t = 0.0:0.1:1.0 f(x) = 2x f2(x) = 3x dx = DiffEqArray([[f(x), f2(x)] for x in t], - t; + t, + [1.0, 2.0]; variables = [:a, :b], + parameters = [:p, :q], independent_variables = [:t]) @test dx[:t] == t @test dx[:a] == [f(x) for x in t] -@test dx[:b] == [f2(x) for x in t] - -dx = DiffEqArray([[f(x), f2(x)] for x in t], - t; - variables = [:a, :b], - independent_variables = [:t]) +@test dx[:a, 2] ≈ f(t[2]) +@test dx[:b, 3] ≈ f2(t[3]) +@test dx[:a, 2:4] ≈ [f(x) for x in t[2:4]] +@test dx[:b, 4:6] ≈ [f2(x) for x in t[4:6]] +@test dx[:b] ≈ [f2(x) for x in t] +@test dx[[:a, :b]] ≈ [[f(x), f2(x)] for x in t] +@test dx[(:a, :b)] == [(f(x), f2(x)) for x in t] +@test dx[[:a, :b], 3] ≈ [f(t[3]), f2(t[3])] +@test dx[[:a, :b], 4:5] ≈ vcat(f.(t[4:5])', f2.(t[4:5])') +@test dx.ps[[:p, :q]] == [1.0, 2.0] +@test dx.ps[:p] == 1.0 +@test dx.ps[:q] == 2.0 @test dx[:t] == t + +@test symbolic_container(dx) isa SymbolCache +@test parameter_values(dx) == [1.0, 2.0] +@test is_variable.((dx,), [:a, :b, :p, :q, :t]) == [true, true, false, false, false] +@test variable_index.((dx,), [:a, :b, :p, :q, :t]) == [1, 2, nothing, nothing, nothing] +@test is_parameter.((dx,), [:a, :b, :p, :q, :t]) == [false, false, true, true, false] +@test parameter_index.((dx,), [:a, :b, :p, :q, :t]) == [nothing, nothing, 1, 2, nothing] +@test is_independent_variable.((dx,), [:a, :b, :p, :q, :t]) == [false, false, false, false, true] +@test variable_symbols(dx) == [:a, :b] +@test parameter_symbols(dx) == [:p, :q] +@test independent_variable_symbols(dx) == [:t] +@test is_time_dependent(dx) +@test constant_structure(dx) + dx = DiffEqArray([[f(x), f2(x)] for x in t], t; variables = [:a, :b]) @test_throws Exception dx[nothing] # make sure it isn't storing [nothing] as indepsym @@ -24,3 +46,4 @@ ABC = @SLVector (:a, :b, :c); A = ABC(1, 2, 3); B = RecursiveArrayTools.DiffEqArray([A, A], [0.0, 2.0]); @test getindex(B, :a) == [1, 1] + From 26c5873762c6bafa94c0c8110a7b3e9b89df40db Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Tue, 28 Nov 2023 14:37:08 +0530 Subject: [PATCH 41/45] refactor: use new getp and setp for parameter indexing --- src/vector_of_array.jl | 15 ++++----------- test/downstream/symbol_indexing.jl | 4 ++-- test/symbolic_indexing_interface_test.jl | 8 +++++--- 3 files changed, 11 insertions(+), 16 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index c08eee2d..4b9d8b94 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -198,14 +198,6 @@ function DiffEqArray(vec::AbstractVector{VT}, sys) end -function Base.getproperty(A::AbstractDiffEqArray, sym::Symbol) - if sym == SymbolicIndexingInterface.PARAMETER_INDEXING_PROXY_PROPERTY_NAME - return ParameterIndexingProxy(A) - else - return getfield(A, sym) - end -end - SymbolicIndexingInterface.parameter_values(A::AbstractDiffEqArray) = A.p SymbolicIndexingInterface.symbolic_container(A::AbstractDiffEqArray) = A.sys @@ -292,8 +284,8 @@ Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray, ::Scalar return getindex.(A.u, variable_index.((A,), (sym,), eachindex(A.t))) end elseif is_parameter(A, sym) - Base.depwarn("Indexing with parameters is deprecated. Use `$(nameof(typeof(A))).$(SymbolicIndexingInterface.PARAMETER_INDEXING_PROXY_PROPERTY_NAME)[$sym]` for parameter indexing.", :parameter_getindex) - A.ps[sym] + Base.depwarn("Indexing with parameters is deprecated. Use `getp(A, $sym)` for parameter indexing.", :parameter_getindex) + return getp(A, sym)(A) elseif is_observed(A, sym) return observed(A, sym).(A.u, (parameter_values(A),), A.t) else @@ -333,7 +325,8 @@ end Base.@propagate_inbounds function Base.getindex(A::AbstractDiffEqArray, ::ScalarSymbolic, sym::Union{Tuple,AbstractArray}) if all(x -> is_parameter(A, x), sym) - return getindex.((A,), sym) + Base.depwarn("Indexing with parameters is deprecated. Use `getp(A, $sym)` for parameter indexing.", :parameter_getindex) + return getp(A, sym)(A) else return [getindex.((A,), sym, i) for i in eachindex(A.t)] end diff --git a/test/downstream/symbol_indexing.jl b/test/downstream/symbol_indexing.jl index 3c79cc26..29ffdd21 100644 --- a/test/downstream/symbol_indexing.jl +++ b/test/downstream/symbol_indexing.jl @@ -1,4 +1,4 @@ -using RecursiveArrayTools, ModelingToolkit, OrdinaryDiffEq, Test +using RecursiveArrayTools, ModelingToolkit, OrdinaryDiffEq, SymbolicIndexingInterface, Test include("../testutils.jl") @@ -21,7 +21,7 @@ sol_new = DiffEqArray(sol.u[1:10], @test sol_new[RHS] ≈ (1 .- sol_new[x]) ./ 3.0 @test sol_new[t] ≈ sol_new.t @test sol_new[t, 1:5] ≈ sol_new.t[1:5] -@test sol.ps[τ] == sol_new.ps[τ] == 3.0 +@test getp(sol, τ)(sol) == getp(sol_new, τ)(sol_new) == 3.0 @test_deprecated sol[τ] @test_deprecated sol_new[τ] diff --git a/test/symbolic_indexing_interface_test.jl b/test/symbolic_indexing_interface_test.jl index 66e2f697..67c3b3b4 100644 --- a/test/symbolic_indexing_interface_test.jl +++ b/test/symbolic_indexing_interface_test.jl @@ -21,9 +21,11 @@ dx = DiffEqArray([[f(x), f2(x)] for x in t], @test dx[(:a, :b)] == [(f(x), f2(x)) for x in t] @test dx[[:a, :b], 3] ≈ [f(t[3]), f2(t[3])] @test dx[[:a, :b], 4:5] ≈ vcat(f.(t[4:5])', f2.(t[4:5])') -@test dx.ps[[:p, :q]] == [1.0, 2.0] -@test dx.ps[:p] == 1.0 -@test dx.ps[:q] == 2.0 +@test getp(dx, [:p, :q])(dx) == [1.0, 2.0] +@test getp(dx, :p)(dx) == 1.0 +@test getp(dx, :q)(dx) == 2.0 +@test_deprecated dx[:p] +@test_deprecated dx[[:p, :q]] @test dx[:t] == t @test symbolic_container(dx) isa SymbolCache From 22a81f21d6149be9dc76b8e45f30faef4a171a96 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Wed, 29 Nov 2023 17:34:21 +0530 Subject: [PATCH 42/45] refactor: revert eachindex to iterating over indices of VA.u --- src/vector_of_array.jl | 12 +++--------- 1 file changed, 3 insertions(+), 9 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 4b9d8b94..2f65c44a 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -206,7 +206,7 @@ Base.IndexStyle(::Type{<:AbstractVectorOfArray}) = IndexCartesian() @inline Base.length(VA::AbstractVectorOfArray) = length(VA.u) @inline function Base.eachindex(VA::AbstractVectorOfArray) - return Iterators.flatten((CartesianIndex(i, j) for i in eachindex(arr)) for (j, arr) in enumerate(VA.u)) + return eachindex(VA.u) end @inline Base.IteratorSize(::Type{<:AbstractVectorOfArray}) = Base.HasLength() @inline Base.first(VA::AbstractVectorOfArray) = first(VA.u) @@ -526,14 +526,8 @@ end Base.reshape(A::VectorOfArray, dims...) = Base.reshape(Array(A), dims...) # Need this for ODE_DEFAULT_UNSTABLE_CHECK from DiffEqBase to work properly -@inline Base.any(f, VA::AbstractVectorOfArray) = any(f, VA[i] for i in eachindex(VA)) -@inline Base.all(f, VA::AbstractVectorOfArray) = all(f, VA[i] for i in eachindex(VA)) -@inline function Base.any(f::Function, VA::AbstractVectorOfArray) - any(f, VA[i] for i in eachindex(VA)) -end -@inline function Base.all(f::Function, VA::AbstractVectorOfArray) - all(f, VA[i] for i in eachindex(VA)) -end +@inline Base.any(f, VA::AbstractVectorOfArray) = any(any(f, u) for u in VA.u) +@inline Base.all(f, VA::AbstractVectorOfArray) = all(all(f, u) for u in VA.u) # conversion tools vecarr_to_vectors(VA::AbstractVectorOfArray) = [VA[i, :] for i in eachindex(VA[1])] From 7b2613c731c43f3af7ce12921dac944dcd3f2331 Mon Sep 17 00:00:00 2001 From: Aayush Sabharwal Date: Fri, 1 Dec 2023 10:36:06 +0530 Subject: [PATCH 43/45] refactor: add Base.view for AbstractVectorOfArray --- src/vector_of_array.jl | 18 ++++++++++++++++-- 1 file changed, 16 insertions(+), 2 deletions(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index 2f65c44a..c67170c7 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -445,6 +445,13 @@ function Base.append!(VA::AbstractVectorOfArray{T, N}, end # AbstractArray methods +function Base.view(A::AbstractVectorOfArray, I::Vararg{Any,M}) where {M} + @inline + J = map(i->Base.unalias(A,i), to_indices(A, I)) + @boundscheck checkbounds(A, J...) + SubArray(IndexStyle(A), A, J, Base.index_dimsum(J...)) +end +Base.check_parent_index_match(::RecursiveArrayTools.AbstractVectorOfArray{T,N}, ::NTuple{N,Bool}) where {T,N} = nothing Base.ndims(::AbstractVectorOfArray{T, N}) where {T, N} = N function Base.checkbounds(::Type{Bool}, VA::AbstractVectorOfArray, idx...) if checkbounds(Bool, VA.u, last(idx)) @@ -456,6 +463,9 @@ function Base.checkbounds(::Type{Bool}, VA::AbstractVectorOfArray, idx...) end return false end +function Base.checkbounds(VA::AbstractVectorOfArray, idx...) + checkbounds(Bool, VA, idx...) || throw(BoundsError(VA, idx)) +end # Operations function Base.isapprox(A::AbstractVectorOfArray, @@ -502,8 +512,12 @@ end # Tools for creating similar objects Base.eltype(::VectorOfArray{T}) where {T} = T # TODO: Is there a better way to do this? -@inline function Base.similar(VA::VectorOfArray, args...) - return Base.similar(ones(eltype(VA)), args...) +@inline function Base.similar(VA::AbstractVectorOfArray, args...) + if args[end] isa Type + return Base.similar(eltype(VA)[], args..., size(VA)) + else + return Base.similar(eltype(VA)[], args...) + end end @inline function Base.similar(VA::VectorOfArray, ::Type{T} = eltype(VA)) where {T} VectorOfArray([similar(VA[:, i], T) for i in eachindex(VA.u)]) From 59f4b86888e2807afa0ffab7bd7c168f5e694477 Mon Sep 17 00:00:00 2001 From: Christopher Rackauckas Date: Mon, 11 Dec 2023 15:31:57 -0500 Subject: [PATCH 44/45] Update src/vector_of_array.jl --- src/vector_of_array.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index c67170c7..a0aa6170 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -217,7 +217,7 @@ function Base.firstindex(VA::AbstractVectorOfArray) end function Base.lastindex(VA::AbstractVectorOfArray) - Base.depwarn("Linear indexing of `AbstractVectorOfArray` is deprecated", :lastindex) + Base.depwarn("Linear indexing of `AbstractVectorOfArray` is deprecated. Change `A[i]` to `A.u[i]` ", :lastindex) return lastindex(VA.u) end From 0936086cc2b5b8249fe907d1f1ff98c778b268af Mon Sep 17 00:00:00 2001 From: Christopher Rackauckas Date: Mon, 11 Dec 2023 15:32:04 -0500 Subject: [PATCH 45/45] Update src/vector_of_array.jl --- src/vector_of_array.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/vector_of_array.jl b/src/vector_of_array.jl index a0aa6170..42560d6b 100644 --- a/src/vector_of_array.jl +++ b/src/vector_of_array.jl @@ -212,7 +212,7 @@ end @inline Base.first(VA::AbstractVectorOfArray) = first(VA.u) @inline Base.last(VA::AbstractVectorOfArray) = last(VA.u) function Base.firstindex(VA::AbstractVectorOfArray) - Base.depwarn("Linear indexing of `AbstractVectorOfArray` is deprecated", :firstindex) + Base.depwarn(Linear indexing of `AbstractVectorOfArray` is deprecated. Change `A[i]` to `A.u[i]` ",, :firstindex) return firstindex(VA.u) end