From 86b8285ac2ae048f4dd7a6deb48c9b843721b31b Mon Sep 17 00:00:00 2001 From: Seth Axen Date: Wed, 1 Feb 2023 22:13:19 +0100 Subject: [PATCH] Make DynamicHMC and Turing support into extensions (#114) * Move integrations to ext folder * Make extensions modules * Use Requires for older Julia versions * Always load Requires * Specify weakdeps and extensions * Increment patch number * Run CI on nightly as well * Only run with 2 threads on latest version * Revert "Only run with 2 threads on latest version" This reverts commit 1b840070217bdc539e73c98fd9cefaeb04524ccc. * Revert "Run CI on nightly as well" This reverts commit 1646855c4adb76545f46e46f4f430737f8c55f4f. * Simplify job matrix * Run DynamicHMC and Turing integration tests on nightly * Load modules using syntax required by Requires * Add Julia version to name * Require Turing dependencies used * Use double-dot syntax only for conditionally loaded deps * Increment version number --- .github/workflows/IntegrationTests.yml | 13 +++++---- Project.toml | 19 +++++++++++- ext/DynamicHMCExt.jl | 22 ++++++++++++++ src/integration/turing.jl => ext/TuringExt.jl | 29 ++++++++++++++----- src/Pathfinder.jl | 16 ++++++---- src/integration/dynamichmc.jl | 11 ------- 6 files changed, 81 insertions(+), 29 deletions(-) create mode 100644 ext/DynamicHMCExt.jl rename src/integration/turing.jl => ext/TuringExt.jl (86%) delete mode 100644 src/integration/dynamichmc.jl diff --git a/.github/workflows/IntegrationTests.yml b/.github/workflows/IntegrationTests.yml index c823bba1..a617b994 100644 --- a/.github/workflows/IntegrationTests.yml +++ b/.github/workflows/IntegrationTests.yml @@ -6,24 +6,27 @@ on: pull_request: jobs: test: - name: ${{ matrix.package }} - runs-on: ${{ matrix.os }} + name: ${{ matrix.package }} - Julia ${{ matrix.version }} + runs-on: ubuntu-latest strategy: fail-fast: false matrix: version: [1] - os: [ubuntu-latest] - arch: [x64] package: - DynamicHMC - AdvancedHMC - Turing + include: + - package: DynamicHMC + version: 'nightly' + - package: Turing + version: 'nightly' steps: - uses: actions/checkout@v2 - uses: julia-actions/setup-julia@v1 with: version: ${{ matrix.version }} - arch: ${{ matrix.arch }} + arch: x64 - uses: julia-actions/julia-buildpkg@v1 - run: | julia --code-coverage=user -e ' diff --git a/Project.toml b/Project.toml index c18c60dd..f3ba6d17 100644 --- a/Project.toml +++ b/Project.toml @@ -1,7 +1,7 @@ name = "Pathfinder" uuid = "b1d3bc72-d0e7-4279-b92f-7fa5d6d2d454" authors = ["Seth Axen and contributors"] -version = "0.7.0" +version = "0.7.1" [deps] Accessors = "7d9f7c33-5ae7-4f3b-8dc6-eff91059b697" @@ -25,13 +25,25 @@ StatsBase = "2913bbd2-ae8a-5f71-8c99-4fb6c76f3a91" Transducers = "28d57a85-8fef-5791-bfe6-a80928e7c999" UnPack = "3a884ed6-31ef-47d7-9d2a-63182c4928ed" +[weakdeps] +DynamicHMC = "bbc10e6e-7c05-544b-b16e-64fede858acb" +DynamicPPL = "366bfd00-2699-11ea-058f-f148b4cae6d8" +MCMCChains = "c7f686f2-ff18-58e9-bc7b-31028e88f75d" +Turing = "fce5fe82-541a-59a6-adf8-730c64b5f9a0" + +[extensions] +DynamicHMCExt = "DynamicHMC" +TuringExt = ["DynamicPPL", "MCMCChains", "Turing"] + [compat] Accessors = "0.1" Distributions = "0.25" +DynamicPPL = "0.20, 0.21" Folds = "0.2" ForwardDiff = "0.10" IrrationalConstants = "0.1.1" LogDensityProblems = "2" +MCMCChains = "5" Optim = "1.4" Optimization = "3" OptimizationOptimJL = "0.1" @@ -42,12 +54,17 @@ Requires = "1" SciMLBase = "1.8.1" StatsBase = "0.33" Transducers = "0.4.5" +Turing = "0.21, 0.22, 0.23" UnPack = "1" julia = "1.6" [extras] +DynamicHMC = "bbc10e6e-7c05-544b-b16e-64fede858acb" +DynamicPPL = "366bfd00-2699-11ea-058f-f148b4cae6d8" +MCMCChains = "c7f686f2-ff18-58e9-bc7b-31028e88f75d" OptimizationNLopt = "4e6fcdb7-1186-4e1f-a706-475e75c168bb" Test = "8dfed614-e22c-5e08-85e1-65c5234f0b40" +Turing = "fce5fe82-541a-59a6-adf8-730c64b5f9a0" [targets] test = ["OptimizationNLopt", "Test"] diff --git a/ext/DynamicHMCExt.jl b/ext/DynamicHMCExt.jl new file mode 100644 index 00000000..b1681e8b --- /dev/null +++ b/ext/DynamicHMCExt.jl @@ -0,0 +1,22 @@ +module DynamicHMCExt + +using PDMats: PDMats +if isdefined(Base, :get_extension) + using Pathfinder: Pathfinder + using DynamicHMC: DynamicHMC +else # using Requires + using ..Pathfinder: Pathfinder + using ..DynamicHMC: DynamicHMC +end + +function DynamicHMC.GaussianKineticEnergy(M⁻¹::Pathfinder.WoodburyPDMat) + return DynamicHMC.GaussianKineticEnergy(M⁻¹, inv(Pathfinder.pdfactorize(M⁻¹).R)) +end + +function DynamicHMC.kinetic_energy( + κ::DynamicHMC.GaussianKineticEnergy{<:Pathfinder.WoodburyPDMat}, p, q=nothing +) + return PDMats.quad(κ.M⁻¹, p) / 2 +end + +end # module diff --git a/src/integration/turing.jl b/ext/TuringExt.jl similarity index 86% rename from src/integration/turing.jl rename to ext/TuringExt.jl index 0cbc05c4..ac2b1f98 100644 --- a/src/integration/turing.jl +++ b/ext/TuringExt.jl @@ -1,5 +1,18 @@ -using .Turing: Turing, DynamicPPL, MCMCChains +module TuringExt + using Accessors: Accessors +using Random: Random +if isdefined(Base, :get_extension) + using DynamicPPL: DynamicPPL + using MCMCChains: MCMCChains + using Pathfinder: Pathfinder + using Turing: Turing +else # using Requires + using ..DynamicPPL: DynamicPPL + using ..MCMCChains: MCMCChains + using ..Pathfinder: Pathfinder + using ..Turing: Turing +end # utilities for working with Turing model parameter names using only the DynamicPPL API @@ -119,30 +132,30 @@ function varnames_to_ranges(metadata::DynamicPPL.Metadata) return Dict(zip(metadata.vns, ranges)) end -function pathfinder( +function Pathfinder.pathfinder( model::DynamicPPL.Model; rng=Random.GLOBAL_RNG, init_scale=2, - init_sampler=UniformSampler(init_scale), + init_sampler=Pathfinder.UniformSampler(init_scale), init=nothing, kwargs..., ) var_names = flattened_varnames_list(model) prob = Turing.optim_problem(model, Turing.MAP(); constrained=false, init_theta=init) init_sampler(rng, prob.prob.u0) - result = pathfinder(prob.prob; rng, input=model, kwargs...) + result = Pathfinder.pathfinder(prob.prob; rng, input=model, kwargs...) draws = reduce(vcat, transpose.(prob.transform.(eachcol(result.draws)))) chns = MCMCChains.Chains(draws, var_names; info=(; pathfinder_result=result)) result_new = Accessors.@set result.draws_transformed = chns return result_new end -function multipathfinder( +function Pathfinder.multipathfinder( model::DynamicPPL.Model, ndraws::Int; rng=Random.GLOBAL_RNG, init_scale=2, - init_sampler=UniformSampler(init_scale), + init_sampler=Pathfinder.UniformSampler(init_scale), nruns::Int, kwargs..., ) @@ -153,9 +166,11 @@ function multipathfinder( for _ in 2:nruns push!(init, init_sampler(rng, deepcopy(init1))) end - result = multipathfinder(fun.func, ndraws; rng, input=model, init, kwargs...) + result = Pathfinder.multipathfinder(fun.func, ndraws; rng, input=model, init, kwargs...) draws = reduce(vcat, transpose.(fun.transform.(eachcol(result.draws)))) chns = MCMCChains.Chains(draws, var_names; info=(; pathfinder_result=result)) result_new = Accessors.@set result.draws_transformed = chns return result_new end + +end # module diff --git a/src/Pathfinder.jl b/src/Pathfinder.jl index 2fec765a..912558eb 100644 --- a/src/Pathfinder.jl +++ b/src/Pathfinder.jl @@ -45,14 +45,20 @@ include("singlepath.jl") include("multipath.jl") function __init__() - Requires.@require DynamicHMC = "bbc10e6e-7c05-544b-b16e-64fede858acb" begin - include("integration/dynamichmc.jl") - end Requires.@require AdvancedHMC = "0bf59076-c3b1-5ca4-86bd-e02cd72cde3d" begin include("integration/advancedhmc.jl") end - Requires.@require Turing = "fce5fe82-541a-59a6-adf8-730c64b5f9a0" begin - include("integration/turing.jl") + @static if !isdefined(Base, :get_extension) + Requires.@require DynamicHMC = "bbc10e6e-7c05-544b-b16e-64fede858acb" begin + include("../ext/DynamicHMCExt.jl") + end + Requires.@require Turing = "fce5fe82-541a-59a6-adf8-730c64b5f9a0" begin + Requires.@require DynamicPPL = "366bfd00-2699-11ea-058f-f148b4cae6d8" begin + Requires.@require MCMCChains = "c7f686f2-ff18-58e9-bc7b-31028e88f75d" begin + include("../ext/TuringExt.jl") + end + end + end end end diff --git a/src/integration/dynamichmc.jl b/src/integration/dynamichmc.jl deleted file mode 100644 index 403ea496..00000000 --- a/src/integration/dynamichmc.jl +++ /dev/null @@ -1,11 +0,0 @@ -using .DynamicHMC: DynamicHMC - -function DynamicHMC.GaussianKineticEnergy(M⁻¹::WoodburyPDMat) - return DynamicHMC.GaussianKineticEnergy(M⁻¹, inv(pdfactorize(M⁻¹).R)) -end - -function DynamicHMC.kinetic_energy( - κ::DynamicHMC.GaussianKineticEnergy{<:WoodburyPDMat}, p, q=nothing -) - return PDMats.quad(κ.M⁻¹, p) / 2 -end