From a8b2bd5b0aae25d529bdf6df97ac8776a160e9ad Mon Sep 17 00:00:00 2001 From: nHackel Date: Thu, 1 Aug 2024 13:34:01 +0200 Subject: [PATCH 1/2] Refactor planpath function to allow specific plan file paths --- src/AlgorithmInterface.jl | 20 ++++++++++++-------- 1 file changed, 12 insertions(+), 8 deletions(-) diff --git a/src/AlgorithmInterface.jl b/src/AlgorithmInterface.jl index cc611a5..591a8f6 100644 --- a/src/AlgorithmInterface.jl +++ b/src/AlgorithmInterface.jl @@ -46,13 +46,17 @@ export planpath, plandir plandir() = abspath(homedir(), ".mpi", "RecoPlans") function planpath(name::AbstractString) + if isfile(name) + return name + end + for dir in [plandir(), joinpath(@__DIR__, "..", "config")] filename = joinpath(dir, string(name, ".toml")) if isfile(filename) return filename end end - throw(ArgumentError("Could not find a suitable MPI reconstruction plan with name $name.\nCustom plans can be stored in $(plandir()).")) + throw(ArgumentError("Could not find a suitable MPI reconstruction plan with name $name. Custom plans can be stored in $(plandir()).")) end const recoPlans = LRU{UInt64, RecoPlan}(maxsize = 3) @@ -65,7 +69,7 @@ Perform a reconstruction with the `RecoPlan` specified by `name` and given `data Additional keyword arguments can be passed to the reconstruction plan. `RecoPlans` can be stored in the directory `$(plandir())` or in the MPIReco package config folder. The first plan found is used. The cache considers the last modification time of the plan file. -If a keyword argument changes the structure of the plan the cache is bypassed. +If a keyword argument changes the structure of the plan the cache is bypassed. Alternatively, name can be a path to specific plan file. The cache can be emptied with `emptyRecoCache!()`. @@ -76,26 +80,26 @@ julia> mdf = MPIFile("data.mdf"); julia> reconstruct("SinglePatch", mdf; solver = Kaczmarz, reg = [L2Regularization(0.3f0)], iterations = 10, frames = 1:10, ...) ``` """ -function reconstruct(name::AbstractString, data::MPIFile, cache::Bool = true; kwargs...) - plan = loadRecoPlan(name, cache; kwargs...) +function reconstruct(name::AbstractString, data::MPIFile, cache::Bool = true, modules = [AbstractImageReconstruction, MPIFiles, MPIReco, RegularizedLeastSquares]; kwargs...) + plan = loadRecoPlan(name, cache, modules; kwargs...) setAll!(plan; kwargs...) return reconstruct(build(plan), data) end -function loadRecoPlan(name::AbstractString, cache::Bool; kwargs...) +function loadRecoPlan(name::AbstractString, cache::Bool, modules; kwargs...) planfile = AbstractImageReconstruction.planpath(MPIReco, name) # If the user disables caching or changes the plan structure we bypass the cache kwargValues = values(values(kwargs)) if !cache || any(val -> isa(val, RecoPlan) || isa(val, AbstractImageReconstructionParameters), kwargValues) - return loadRecoPlan(planfile) + return loadRecoPlan(planfile, modules) end key = hash(planfile, hash(mtime(planfile))) return get!(recoPlans, key) do - loadRecoPlan(planfile) + loadRecoPlan(planfile, modules) end end -loadRecoPlan(planfile::AbstractString) = loadPlan(planfile, [AbstractImageReconstruction, MPIFiles, MPIReco, RegularizedLeastSquares]) +loadRecoPlan(planfile::AbstractString, modules) = loadPlan(planfile, modules) export emptyRecoCache! """ From 928788e8d7dbd3d085d1d9330b637d3acb14de1b Mon Sep 17 00:00:00 2001 From: nHackel Date: Thu, 1 Aug 2024 14:01:00 +0200 Subject: [PATCH 2/2] Composite Paramters for now are vectors of abstract types, otherwise serialization breaks --- src/PreProcessing/FrequencyFilterParameter.jl | 19 ++++++++++++++----- src/Weighting.jl | 4 ++-- 2 files changed, 16 insertions(+), 7 deletions(-) diff --git a/src/PreProcessing/FrequencyFilterParameter.jl b/src/PreProcessing/FrequencyFilterParameter.jl index 3bbac0a..ee7e3e0 100644 --- a/src/PreProcessing/FrequencyFilterParameter.jl +++ b/src/PreProcessing/FrequencyFilterParameter.jl @@ -10,10 +10,19 @@ function process(::Type{<:AbstractMPIRecoAlgorithm}, params::NoFrequencyFilterPa end export DirectSelectionFrequencyFilterParameters -Base.@kwdef struct DirectSelectionFrequencyFilterParameters{T <: Union{Integer, CartesianIndex{2}}, FIT <: AbstractVector{T}} <: AbstractFrequencyFilterParameter - freqIndices::Union{Nothing, FIT} = nothing +struct DirectSelectionFrequencyFilterParameters{T <: Union{Integer, CartesianIndex{2}}, FIT <: AbstractVector{T}} <: AbstractFrequencyFilterParameter + freqIndices::Union{Nothing, FIT} + function DirectSelectionFrequencyFilterParameters(;freqIndices::FIT = nothing) where FIT + if isnothing(freqIndices) + el = CartesianIndex{2} + v = Vector{el} + else + el = eltype(freqIndices) + v = typeof(v) + end + new{el, v}(freqIndices) + end end -DirectSelectionFrequencyFilterParameters() = DirectSelectionFrequencyFilterParameters{CartesianIndex{2}, Vector{CartesianIndex{2}}}(nothing) function process(::Type{<:AbstractMPIRecoAlgorithm}, params::DirectSelectionFrequencyFilterParameters{T}, file::MPIFile) where T <: Integer nFreq = params.freqIndices nReceivers = rxNumChannels(file) @@ -71,8 +80,8 @@ function process(::Type{<:AbstractMPIRecoAlgorithm}, params::AbstractFrequencyFi end export CompositeFrequencyFilterParameters -Base.@kwdef struct CompositeFrequencyFilterParameters{FS} <: AbstractFrequencyFilterParameter where FS <: AbstractFrequencyFilterParameter - filters::Vector{FS} +Base.@kwdef struct CompositeFrequencyFilterParameters <: AbstractFrequencyFilterParameter + filters::Vector{AbstractFrequencyFilterParameter} end function process(algoT::Type{<:AbstractMPIRecoAlgorithm}, params::CompositeFrequencyFilterParameters, file::MPIFile) return reduce(intersect, filter(!isnothing, map(p -> process(algoT, p, file), params.filters))) diff --git a/src/Weighting.jl b/src/Weighting.jl index a977488..51e9ba9 100644 --- a/src/Weighting.jl +++ b/src/Weighting.jl @@ -49,8 +49,8 @@ function process(::Type{<:AbstractMPIRecoAlgorithm}, params::RowNormWeightingPar end export CompositeWeightingParameters -Base.@kwdef struct CompositeWeightingParameters{WS} <: AbstractWeightingParameters where WS <: AbstractWeightingParameters - weightingParameters::Vector{WS} +Base.@kwdef struct CompositeWeightingParameters <: AbstractWeightingParameters + weightingParameters::Vector{AbstractWeightingParameters} end function process(algoT::Type{<:AbstractMPIRecoAlgorithm}, params::CompositeWeightingParameters, args...) weights = map(p -> process(algoT, p, args...), params.weightingParameters)