using LogDensityProblems: LogDensityProblems;
# Let's define some type that represents the model.
struct IsotropicNormalModel{M<:AbstractVector{<:Real}}
"mean of the isotropic Gaussian"
::M
meanend
# Specifies what input length the model expects.
dimension(model::IsotropicNormalModel) = length(model.mean)
LogDensityProblems.# Implementation of the log-density evaluation of the model.
function LogDensityProblems.logdensity(model::IsotropicNormalModel, x::AbstractVector{<:Real})
return - sum(abs2, x .- model.mean) / 2
end
Implementing samplers
In this tutorial, we’ll go through step-by-step how to implement a “simple” sampler in AbstractMCMC.jl in such a way that it can be easily applied to Turing.jl models.
In particular, we’re going to implement a version of Metropolis-adjusted Langevin (MALA).
Note that we will implement this sampler in the AbstractMCMC.jl framework, completely “ignoring” Turing.jl until the very end of the tutorial, at which point we’ll use a single line of code to make the resulting sampler available to Turing.jl. This is to really drive home the point that one can implement samplers in a way that is accessible to all of Turing.jl’s users without having to use Turing.jl yourself.
Quick overview of MALA
We can view MALA as a single step of the leapfrog intergrator with resampling of momentum \(p\) at every step.1 To make that statement a bit more concrete, we first define the extended target \(\bar{\gamma}(x, p)\) as
\[\begin{equation*} \log \bar{\gamma}(x, p) \propto \log \gamma(x) + \log \gamma_{\mathcal{N}(0, M)}(p) \end{equation*}\]
where \(\gamma_{\mathcal{N}(0, M)}\) denotes the density for a zero-centered Gaussian with covariance matrix \(M\). We then consider targeting this joint distribution over both \(x\) and \(p\) as follows. First we define the map
\[\begin{equation*} \begin{split} L_{\epsilon}: \quad & \mathbb{R}^d \times \mathbb{R}^d \to \mathbb{R}^d \times \mathbb{R}^d \\ & (x, p) \mapsto (\tilde{x}, \tilde{p}) := L_{\epsilon}(x, p) \end{split} \end{equation*}\]
as
\[\begin{equation*} \begin{split} p_{1 / 2} &:= p + \frac{\epsilon}{2} \nabla \log \gamma(x) \\ \tilde{x} &:= x + \epsilon M^{-1} p_{1 /2 } \\ p_1 &:= p_{1 / 2} + \frac{\epsilon}{2} \nabla \log \gamma(\tilde{x}) \\ \tilde{p} &:= - p_1 \end{split} \end{equation*}\]
This might be familiar for some readers as a single step of the Leapfrog integrator. We then define the MALA kernel as follows: given the current iterate \(x_i\), we sample the next iterate \(x_{i + 1}\) as
\[\begin{equation*} \begin{split} p &\sim \mathcal{N}(0, M) \\ (\tilde{x}, \tilde{p}) &:= L_{\epsilon}(x_i, p) \\ \alpha &:= \min \left\{ 1, \frac{\bar{\gamma}(\tilde{x}, \tilde{p})}{\bar{\gamma}(x_i, p)} \right\} \\ x_{i + 1} &:= \begin{cases} \tilde{x} \quad & \text{ with prob. } \alpha \\ x_i \quad & \text{ with prob. } 1 - \alpha \end{cases} \end{split} \end{equation*}\]
i.e. we accept the proposal \(\tilde{x}\) with probability \(\alpha\) and reject it, thus sticking with our current iterate, with probability \(1 - \alpha\).
What we need from a model: LogDensityProblems.jl
There are a few things we need from the “target” / “model” / density that we want to sample from:
- We need access to log-density evaluations \(\log \gamma(x)\) so we can compute the acceptance ratio involving \(\log \bar{\gamma}(x, p)\).
- We need access to log-density gradients \(\nabla \log \gamma(x)\) so we can compute the Leapfrog steps \(L_{\epsilon}(x, p)\).
- We also need access to the “size” of the model so we can determine the size of \(M\).
Luckily for us, there is a package called LogDensityProblems.jl which provides an interface for exactly this!
To demonstrate how one can implement the “LogDensityProblems.jl interface”2 we will use a simple Gaussian model as an example:
This gives us all of the properties we want for our MALA sampler with the exception of the computation of the gradient \(\nabla \log \gamma(x)\). There is the method LogDensityProblems.logdensity_and_gradient
which should return a 2-tuple where the first entry is the evaluation of the logdensity \(\log \gamma(x)\) and the second entry is the gradient \(\nabla \log \gamma(x)\).
There are two ways to “implement” this method: 1) we implement it by hand, which is feasible in the case of our IsotropicNormalModel
, or b) we defer the implementation of this to a automatic differentiation backend.
To implement it by hand we can simply do
# Tell LogDensityProblems.jl that first-order, i.e. gradient information, is available.
capabilities(model::IsotropicNormalModel) = LogDensityProblems.LogDensityOrder{1}()
LogDensityProblems.
# Implement `logdensity_and_gradient`.
function LogDensityProblems.logdensity_and_gradient(model::IsotropicNormalModel, x)
= LogDensityProblems.logdensity(model, x)
logγ_x = -x .* (x - model.mean)
∇logγ_x return logγ_x, ∇logγ_x
end
Let’s just try it out:
# Instantiate the problem.
= IsotropicNormalModel([-5., 0., 5.])
model # Create some example input that we can test on.
= randn(LogDensityProblems.dimension(model))
x_example # Evaluate!
logdensity(model, x_example) LogDensityProblems.
-29.004462286411766
To defer it to an automatic differentiation backend, we can do
# Tell LogDensityProblems.jl we only have access to 0-th order information.
capabilities(model::IsotropicNormalModel) = LogDensityProblems.LogDensityOrder{0}()
LogDensityProblems.
# Use `LogDensityProblemsAD`'s `ADgradient` in combination with some AD backend to implement `logdensity_and_gradient`.
using LogDensityProblemsAD, ADTypes, ForwardDiff
= ADgradient(AutoForwardDiff(), model)
model_with_grad logdensity(model_with_grad, x_example) LogDensityProblems.
-29.004462286411766
We’ll continue with the second approach in this tutorial since this is typically what one does in practice, because there are better hobbies to spend time on than deriving gradients by hand.
At this point, one might wonder how we’re going to tie this back to Turing.jl in the end. Effectively, when working with inference methods that only require log-density evaluations and / or higher-order information of the log-density, Turing.jl actually converts the user-provided Model
into an object implementing the above methods for LogDensityProblems.jl. As a result, most samplers provided by Turing.jl are actually implemented to work with LogDensityProblems.jl, enabling their use both within Turing.jl and outside of Turing.jl! Morever, there exists similar conversions for Stan through BridgeStan and StanLogDensityProblems.jl, which means that a sampler supporting the LogDensityProblems.jl interface can easily be used on both Turing.jl and Stan models (in addition to user-provided models, as our IsotropicNormalModel
above)!
Anyways, let’s move on to actually implementing the sampler.
Implementing MALA in AbstractMCMC.jl
Now that we’ve established that a model implementing the LogDensityProblems.jl interface provides us with all the information we need from \(\log \gamma(x)\), we can address the question: given an object that implements the LogDensityProblems.jl interface, how can we define a sampler for it?
We’re going to do this by making our sampler a sub-type of AbstractMCMC.AbstractSampler
in addition to implementing a few methods from AbstractMCMC.jl. Why? Because it gets us a lot of functionality for free, as we will see later.
Moreover, AbstractMCMC.jl provides a very natural interface for MCMC algorithms.
First, we’ll define our MALA
type
using AbstractMCMC
struct MALA{T,A} <: AbstractMCMC.AbstractSampler
"stepsize used in the leapfrog step"
::T
ϵ_init"covariance matrix used for the momentum"
::A
M_initend
Notice how we’ve added the suffix _init
to both the stepsize and the covariance matrix. We’ve done this because a AbstractMCMC.AbstractSampler
should be immutable. Of course there might be many scenarios where we want to allow something like the stepsize and / or the covariance matrix to vary between iterations, e.g. during the burn-in / adaptation phase of the sampling process we might want to adjust the parameters using statistics computed from these initial iterations. But information which can change between iterations should not go in the sampler itself! Instead, this information should go in the sampler state.
The sampler state should at the very least contain all the necessary information to perform the next MCMC iteration, but usually contains further information, e.g. quantities and statistics useful for evaluating whether the sampler has converged.
We will use the following sampler state for our MALA
sampler:
struct MALAState{A<:AbstractVector{<:Real}}
"current position"
::A
xend
This might seem overly redundant: we’re defining a type MALAState
and it only contains a simple vector of reals. In this particular case we indeed could have dropped this and simply used a AbstractVector{<:Real}
as our sampler state, but typically, as we will see later, one wants to include other quantities in the sampler state. For example, if we also wanted to adapt the parameters of our MALA
, e.g. alter the stepsize depending on acceptance rates, in which case we should also put ϵ
in the state, but for now we’ll keep things simple.
Moreover, we also want a sample type, which is a type meant for “public consumption”, i.e. the end-user. This is generally going to contain a subset of the information present in the state. But in such a simple scenario as this, we similarly only have a AbstractVector{<:Real}
:
struct MALASample{A<:AbstractVector{<:Real}}
"current position"
::A
xend
We currently have three things:
- A
AbstractMCMC.AbstractSampler
implementation calledMALA
. - A state
MALAState
for our samplerMALA
. - A sample
MALASample
for our samplerMALA
.
That means that we’re ready to implement the only thing that really matters: AbstractMCMC.step
.
AbstractMCMC.step
defines the MCMC iteration of our MALA
given the current MALAState
. Specifically, the signature of the function is as follows:
function AbstractMCMC.step(
# The RNG to ensure reproducibility.
::Random.AbstractRNG,
rng# The model that defines our target.
::AbstractMCMC.AbstractModel,
model# The sampler for which we're taking a `step`.
::AbstractMCMC.AbstractSampler,
sampler# The current sampler `state`.
state;# Additional keyword arguments that we may or may not need.
...
kwargs )
Moreover, there is a specific AbstractMCMC.AbstractModel
which is used to indicate that the model that is provided implements the LogDensityProblems.jl interface: AbstractMCMC.LogDensityModel
.
Since, as we discussed earlier, in our case we’re indeed going to work with types that support the LogDensityProblems.jl interface, we’ll define AbstractMCMC.step
for such a AbstractMCMC.LogDensityModel
.
Note that AbstractMCMC.LogDensityModel
has no other purpose; it has a single field called logdensity
, and it does nothing else. But by wrapping the model in AbstractMCMC.LogDensityModel
, it allows samplers that want to work with LogDensityProblems.jl to define their AbstractMCMC.step
on this type without running into method ambiguities.
All in all, that means that the signature for our AbstractMCMC.step
is going to be the following:
function AbstractMCMC.step(
::Random.AbstractRNG,
rng# `LogDensityModel` so we know we're working with LogDensityProblems.jl model.
::AbstractMCMC.LogDensityModel,
model# Our sampler.
::MALA,
sampler# Our sampler state.
::MALAState;
state...
kwargs )
Great! Now let’s actually implement the full AbstractMCMC.step
for our MALA
.
Let’s remind ourselves what we’re going to do:
- Sample a new momentum \(p\).
- Compute the log-density of the extended target \(\log \bar{\gamma}(x, p)\).
- Take a single leapfrog step \((\tilde{x}, \tilde{p}) = L_{\epsilon}(x, p)\).
- Accept or reject the proposed \((\tilde{x}, \tilde{p})\).
All in all, this results in the following:
using Random: Random
using Distributions # so we get the `MvNormal`
function AbstractMCMC.step(
::Random.AbstractRNG,
rng::AbstractMCMC.LogDensityModel,
model_wrapper::MALA,
sampler::MALAState;
state...
kwargs
)# Extract the wrapped model which implements LogDensityProblems.jl.
= model_wrapper.logdensity
model # Let's just extract the sampler parameters to make our lives easier.
= sampler.ϵ_init
ϵ = sampler.M_init
M # Extract the current parameters.
= state.x
x # Sample the momentum.
= MvNormal(zeros(LogDensityProblems.dimension(model)), M)
p_dist = rand(rng, p_dist)
p # Propose using a single leapfrog step.
= leapfrog_step(model, x, p, ϵ, M)
x̃, p̃ # Accept or reject proposal.
= LogDensityProblems.logdensity(model, x) + logpdf(p_dist, p)
logp = LogDensityProblems.logdensity(model, x̃) + logpdf(p_dist, p̃)
logp̃ = logp̃ - logp
logα = if log(rand(rng)) < logα
state_new # Accept.
MALAState(x̃)
else
# Reject.
MALAState(x)
end
# Return the "sample" and the sampler state.
return MALASample(state_new.x), state_new
end
Fairly straight-forward.
Of course, we haven’t defined the leapfrog_step
method yet, so let’s do that:
function leapfrog_step(model, x, p, ϵ, M)
# Update momentum `p` using "position" `x`.
= last(LogDensityProblems.logdensity_and_gradient(model, x))
∇logγ_x = p + (ϵ / 2) .* ∇logγ_x
p1 # Update the "position" `x` using momentum `p1`.
= x + ϵ .* (M \ p1)
x̃ # Update momentum `p1` using position `x̃`
= last(LogDensityProblems.logdensity_and_gradient(model, x̃))
∇logγ_x̃ = p1 + (ϵ / 2) .* ∇logγ_x̃
p2 # Flip momentum `p2`.
= -p2
p̃ return x̃, p̃
end
leapfrog_step (generic function with 1 method)
With all of this, we’re technically ready to sample!
using Random, LinearAlgebra
= Random.default_rng()
rng = MALA(1, I)
sampler = MALAState(zeros(LogDensityProblems.dimension(model)))
state
= AbstractMCMC.step(
x_next, state_next
rng,LogDensityModel(model),
AbstractMCMC.
sampler,
state )
(MALASample{Vector{Float64}}([0.0, 0.0, 0.0]), MALAState{Vector{Float64}}([0.0, 0.0, 0.0]))
Great, it works!
And I promised we would get quite some functionality for free if we implemented AbstractMCMC.step
, and so we can now simply call sample
to perform standard MCMC sampling:
# Perform 1000 iterations with our `MALA` sampler.
= sample(model_with_grad, sampler, 10_000; initial_state=state, progress=false)
samples # Concatenate into a matrix.
= stack(sample -> sample.x, samples) samples_matrix
3×10000 Matrix{Float64}:
-3.17767 -2.62914 -3.41283 -5.06294 … -6.05257 -6.05257 -4.24818
0.048018 -0.280721 0.0766346 0.316279 1.69975 1.69975 1.15642
3.63518 4.72931 6.7454 5.26646 4.62322 4.62322 4.31014
# Compute the marginal means and standard deviations.
hcat(mean(samples_matrix; dims=2), std(samples_matrix; dims=2))
3×2 Matrix{Float64}:
-4.95581 0.993768
-0.0184816 0.999101
4.99211 1.00667
Let’s visualize the samples
using StatsPlots
plot(transpose(samples_matrix[:, 1:10:end]), alpha=0.5, legend=false)
Look at that! Things are working; amazin’.
We can also exploit AbstractMCMC.jl’s parallel sampling capabilities:
# Run separate 4 chains for 10 000 iterations using threads to parallelize.
= 4
num_chains = sample(
samples
model_with_grad,
sampler,MCMCThreads(),
10_000,
num_chains;# Note we need to provide an initial state for every chain.
=fill(state, num_chains),
initial_state=false
progress
)= stack(map(Base.Fix1(stack, sample -> sample.x), samples)) samples_array
3×10000×4 Array{Float64, 3}:
[:, :, 1] =
-2.37585 -4.25622 -4.93743 … -4.69165 -6.18423 -5.38679
-1.10524 -0.705976 -0.472967 1.08799 1.2657 0.00193171
1.77446 5.03641 4.37656 6.77822 5.07548 4.6944
[:, :, 2] =
-2.60865 -4.50344 -2.46051 … -5.4489 -4.71648 -6.22634
-0.153148 0.127624 0.0554437 -0.401556 -2.07589 -0.262462
4.1194 3.86143 4.83962 5.19894 5.01143 5.73931
[:, :, 3] =
-0.725164 -1.38017 -5.31099 -5.50821 … -6.30186 -6.7478 -4.79135
-0.841622 -1.06817 -1.24007 0.0499901 0.602418 1.29773 0.457246
1.36109 2.40917 4.80617 6.0696 4.87715 4.53798 5.46108
[:, :, 4] =
-1.67405 -3.11401 -4.86046 … -6.13446 -5.67378 -5.18917
-0.701385 -0.416104 -0.662169 0.404531 1.1324 0.844721
1.43097 3.15126 4.45546 5.15057 5.5194 4.7478
But the fact that we have to provide the AbstractMCMC.sample
call, etc. with an initial_state
to get started is a bit annoying. We can avoid this by also defining a AbstractMCMC.step
without the state
argument:
function AbstractMCMC.step(
::Random.AbstractRNG,
rng::AbstractMCMC.LogDensityModel,
model_wrapper::MALA;
# NOTE: No state provided!
...
kwargs
)= model_wrapper.logdensity
model # Let's just create the initial state by sampling using a Gaussian.
= randn(rng, LogDensityProblems.dimension(model))
x
return MALASample(x), MALAState(x)
end
Equipped with this, we no longer need to provide the initial_state
everywhere:
= sample(model_with_grad, sampler, 10_000; progress=false)
samples = stack(sample -> sample.x, samples)
samples_matrix hcat(mean(samples_matrix; dims=2), std(samples_matrix; dims=2))
3×2 Matrix{Float64}:
-4.98021 0.999211
-0.000726335 1.00483
4.97085 0.992142
Using our sampler with Turing.jl
As we promised, all of this hassle of implementing our MALA
sampler in a way that uses LogDensityProblems.jl and AbstractMCMC.jl gets us something more than just an “automatic” implementation of AbstractMCMC.sample
.
It also enables use with Turing.jl through the externalsampler
, but we need to do one final thing first: we need to tell Turing.jl how to extract a vector of parameters from the “sample” returned in our implementation of AbstractMCMC.step
. In our case, the “sample” is a MALASample
, so we just need the following line:
# Load Turing.jl.
using Turing
# Overload the `getparams` method for our "sample" type, which is just a vector.
getparams(::Turing.Model, sample::MALASample) = sample.x Turing.Inference.
And with that, we’re good to go!
# Our previous model defined as a Turing.jl model.
@model mvnormal_model() = x ~ MvNormal([-5., 0., 5.], I)
# Instantiate our model.
= mvnormal_model()
turing_model # Call `sample` but now we're passing in a Turing.jl `model` and wrapping
# our `MALA` sampler in the `externalsampler` to tell Turing.jl that the sampler
# expects something that implements LogDensityProblems.jl.
= sample(turing_model, externalsampler(sampler), 10_000; progress=false) chain
Chains MCMC chain (10000×4×1 Array{Float64, 3}):
Iterations = 1:1:10000
Number of chains = 1
Samples per chain = 10000
Wall duration = 2.74 seconds
Compute duration = 2.74 seconds
parameters = x[1], x[2], x[3]
internals = lp
Summary Statistics
parameters mean std mcse ess_bulk ess_tail rhat ⋯
Symbol Float64 Float64 Float64 Float64 Float64 Float64 ⋯
x[1] -4.9738 1.0005 0.0182 3020.8444 5558.1478 1.0008 ⋯
x[2] 0.0270 1.0156 0.0176 3336.7795 5420.0090 1.0005 ⋯
x[3] 5.0020 0.9975 0.0187 2857.8702 4975.1915 1.0006 ⋯
1 column omitted
Quantiles
parameters 2.5% 25.0% 50.0% 75.0% 97.5%
Symbol Float64 Float64 Float64 Float64 Float64
x[1] -6.9200 -5.6505 -4.9751 -4.2908 -3.0408
x[2] -1.9297 -0.6856 0.0242 0.7251 2.0020
x[3] 3.0758 4.3271 4.9842 5.6787 6.9753
Pretty neat, eh?
Models with constrained parameters
One thing we’ve sort of glossed over in all of the above is that MALA, at least how we’ve implemented it, requires \(x\) to live in \(\mathbb{R}^d\) for some \(d > 0\). If some of the parameters were in fact constrained, e.g. we were working with a Beta
distribution which has support on the interval \((0, 1)\), not on \(\mathbb{R}^d\), we could easily end up outside of the valid range \((0, 1)\).
@model beta_model() = x ~ Beta(3, 3)
= beta_model()
turing_model = sample(turing_model, externalsampler(sampler), 10_000; progress=false) chain
Chains MCMC chain (10000×2×1 Array{Float64, 3}):
Iterations = 1:1:10000
Number of chains = 1
Samples per chain = 10000
Wall duration = 1.86 seconds
Compute duration = 1.86 seconds
parameters = x
internals = lp
Summary Statistics
parameters mean std mcse ess_bulk ess_tail rhat ⋯
Symbol Float64 Float64 Float64 Float64 Float64 Float64 ⋯
x 0.4978 0.1875 0.0028 4401.5286 5995.5262 1.0000 ⋯
1 column omitted
Quantiles
parameters 2.5% 25.0% 50.0% 75.0% 97.5%
Symbol Float64 Float64 Float64 Float64 Float64
x 0.1418 0.3602 0.5000 0.6346 0.8526
Yep, that still works, but only because Turing.jl actually transforms the turing_model
from constrained to unconstrained, so that the sampler
provided to externalsampler
is actually always working in unconstrained space! This is not always desirable, so we can turn this off:
= sample(turing_model, externalsampler(sampler; unconstrained=false), 10_000; progress=false) chain
Chains MCMC chain (10000×2×1 Array{Float64, 3}):
Iterations = 1:1:10000
Number of chains = 1
Samples per chain = 10000
Wall duration = 0.14 seconds
Compute duration = 0.14 seconds
parameters = x
internals = lp
Summary Statistics
parameters mean std mcse ess_bulk ess_tail rhat e ⋯
Symbol Float64 Float64 Float64 Float64 Float64 Float64 ⋯
x 0.1822 0.0000 0.0000 NaN NaN NaN ⋯
1 column omitted
Quantiles
parameters 2.5% 25.0% 50.0% 75.0% 97.5%
Symbol Float64 Float64 Float64 Float64 Float64
x 0.1822 0.1822 0.1822 0.1822 0.1822
The fun thing is that this still sort of works because
logpdf(Beta(3, 3), 10.0)
-Inf
and so the samples that fall outside of the range are always rejected. But do notice how much worse all the diagnostics are, e.g. ess_tail
is very poor compared to when we use unconstrained=true
. Moreover, in more complex cases this won’t just result in a “nice” -Inf
log-density value, but instead will error:
@model function demo()
~ truncated(Normal(), lower=0)
σ² # If we end up with negative values for `σ²`, the `Normal` will error.
~ Normal(0, σ²)
x end
sample(demo(), externalsampler(sampler; unconstrained=false), 10_000; progress=false)
DomainError: DomainError(-0.05942049545162304, "Normal: the condition σ >= zero(σ) is not satisfied.")
DomainError with -0.05942049545162304:
Normal: the condition σ >= zero(σ) is not satisfied.
Stacktrace:
[1] #371
@ ~/.julia/packages/Distributions/j0ZcJ/src/univariate/continuous/normal.jl:37 [inlined]
[2] check_args
@ ~/.julia/packages/Distributions/j0ZcJ/src/utils.jl:89 [inlined]
[3] #Normal#370
@ ~/.julia/packages/Distributions/j0ZcJ/src/univariate/continuous/normal.jl:37 [inlined]
[4] Normal
@ ~/.julia/packages/Distributions/j0ZcJ/src/univariate/continuous/normal.jl:36 [inlined]
[5] Normal
@ ~/.julia/packages/Distributions/j0ZcJ/src/univariate/continuous/normal.jl:42 [inlined]
[6] macro expansion
@ ~/.julia/packages/DynamicPPL/cvlfK/src/compiler.jl:584 [inlined]
[7] demo
@ ~/work/docs/docs/tutorials/docs-17-implementing-samplers/index.qmd:455 [inlined]
[8] _evaluate!!
@ ~/.julia/packages/DynamicPPL/cvlfK/src/model.jl:914 [inlined]
[9] evaluate_threadsafe!!(model::DynamicPPL.Model{typeof(demo), (), (), (), Tuple{}, Tuple{}, DynamicPPL.DefaultContext}, varinfo::DynamicPPL.TypedVarInfo{@NamedTuple{σ²::DynamicPPL.Metadata{Dict{AbstractPPL.VarName{:σ², typeof(identity)}, Int64}, Vector{Truncated{Normal{Float64}, Continuous, Float64, Float64, Nothing}}, Vector{AbstractPPL.VarName{:σ², typeof(identity)}}, Vector{Float64}, Vector{Set{DynamicPPL.Selector}}}, x::DynamicPPL.Metadata{Dict{AbstractPPL.VarName{:x, typeof(identity)}, Int64}, Vector{Normal{Float64}}, Vector{AbstractPPL.VarName{:x, typeof(identity)}}, Vector{Float64}, Vector{Set{DynamicPPL.Selector}}}}, Float64}, context::DynamicPPL.ValuesAsInModelContext{OrderedDict{Any, Any}, DynamicPPL.DefaultContext})
@ DynamicPPL ~/.julia/packages/DynamicPPL/cvlfK/src/model.jl:903
[10] evaluate!!
@ ~/.julia/packages/DynamicPPL/cvlfK/src/model.jl:833 [inlined]
[11] values_as_in_model
@ ~/.julia/packages/DynamicPPL/cvlfK/src/values_as_in_model.jl:196 [inlined]
[12] values_as_in_model
@ ~/.julia/packages/DynamicPPL/cvlfK/src/values_as_in_model.jl:195 [inlined]
[13] getparams(model::DynamicPPL.Model{typeof(demo), (), (), (), Tuple{}, Tuple{}, DynamicPPL.DefaultContext}, vi::DynamicPPL.TypedVarInfo{@NamedTuple{σ²::DynamicPPL.Metadata{Dict{AbstractPPL.VarName{:σ², typeof(identity)}, Int64}, Vector{Truncated{Normal{Float64}, Continuous, Float64, Float64, Nothing}}, Vector{AbstractPPL.VarName{:σ², typeof(identity)}}, Vector{Float64}, Vector{Set{DynamicPPL.Selector}}}, x::DynamicPPL.Metadata{Dict{AbstractPPL.VarName{:x, typeof(identity)}, Int64}, Vector{Normal{Float64}}, Vector{AbstractPPL.VarName{:x, typeof(identity)}}, Vector{Float64}, Vector{Set{DynamicPPL.Selector}}}}, Float64})
@ Turing.Inference ~/.julia/packages/Turing/NQDYt/src/mcmc/Inference.jl:381
[14] Turing.Inference.Transition(model::DynamicPPL.Model{typeof(demo), (), (), (), Tuple{}, Tuple{}, DynamicPPL.DefaultContext}, vi::DynamicPPL.TypedVarInfo{@NamedTuple{σ²::DynamicPPL.Metadata{Dict{AbstractPPL.VarName{:σ², typeof(identity)}, Int64}, Vector{Truncated{Normal{Float64}, Continuous, Float64, Float64, Nothing}}, Vector{AbstractPPL.VarName{:σ², typeof(identity)}}, Vector{Float64}, Vector{Set{DynamicPPL.Selector}}}, x::DynamicPPL.Metadata{Dict{AbstractPPL.VarName{:x, typeof(identity)}, Int64}, Vector{Normal{Float64}}, Vector{AbstractPPL.VarName{:x, typeof(identity)}}, Vector{Float64}, Vector{Set{DynamicPPL.Selector}}}}, Float64}, t::MALASample{Vector{Float64}})
@ Turing.Inference ~/.julia/packages/Turing/NQDYt/src/mcmc/Inference.jl:256
[15] transition_to_turing
@ ~/.julia/packages/Turing/NQDYt/src/mcmc/abstractmcmc.jl:12 [inlined]
[16] transition_to_turing(f::LogDensityProblemsADForwardDiffExt.ForwardDiffLogDensity{LogDensityFunction{DynamicPPL.TypedVarInfo{@NamedTuple{σ²::DynamicPPL.Metadata{Dict{AbstractPPL.VarName{:σ², typeof(identity)}, Int64}, Vector{Truncated{Normal{Float64}, Continuous, Float64, Float64, Nothing}}, Vector{AbstractPPL.VarName{:σ², typeof(identity)}}, Vector{Float64}, Vector{Set{DynamicPPL.Selector}}}, x::DynamicPPL.Metadata{Dict{AbstractPPL.VarName{:x, typeof(identity)}, Int64}, Vector{Normal{Float64}}, Vector{AbstractPPL.VarName{:x, typeof(identity)}}, Vector{Float64}, Vector{Set{DynamicPPL.Selector}}}}, Float64}, DynamicPPL.Model{typeof(demo), (), (), (), Tuple{}, Tuple{}, DynamicPPL.DefaultContext}, Nothing}, ForwardDiff.Chunk{2}, ForwardDiff.Tag{DynamicPPL.DynamicPPLTag, Float64}, ForwardDiff.GradientConfig{ForwardDiff.Tag{DynamicPPL.DynamicPPLTag, Float64}, Float64, 2, Vector{ForwardDiff.Dual{ForwardDiff.Tag{DynamicPPL.DynamicPPLTag, Float64}, Float64, 2}}}}, transition::MALASample{Vector{Float64}})
@ Turing.Inference ~/.julia/packages/Turing/NQDYt/src/mcmc/abstractmcmc.jl:17
[17] step(rng::TaskLocalRNG, model::DynamicPPL.Model{typeof(demo), (), (), (), Tuple{}, Tuple{}, DynamicPPL.DefaultContext}, sampler_wrapper::DynamicPPL.Sampler{Turing.Inference.ExternalSampler{MALA{Int64, UniformScaling{Bool}}, AutoForwardDiff{nothing, Nothing}, false}}; initial_state::Nothing, initial_params::Nothing, kwargs::@Kwargs{})
@ Turing.Inference ~/.julia/packages/Turing/NQDYt/src/mcmc/abstractmcmc.jl:147
[18] macro expansion
@ ~/.julia/packages/AbstractMCMC/FSyVk/src/sample.jl:0 [inlined]
[19] macro expansion
@ ~/.julia/packages/AbstractMCMC/FSyVk/src/logging.jl:16 [inlined]
[20] mcmcsample(rng::TaskLocalRNG, model::DynamicPPL.Model{typeof(demo), (), (), (), Tuple{}, Tuple{}, DynamicPPL.DefaultContext}, sampler::DynamicPPL.Sampler{Turing.Inference.ExternalSampler{MALA{Int64, UniformScaling{Bool}}, AutoForwardDiff{nothing, Nothing}, false}}, N::Int64; progress::Bool, progressname::String, callback::Nothing, num_warmup::Int64, discard_initial::Int64, thinning::Int64, chain_type::Type, initial_state::Nothing, kwargs::@Kwargs{})
@ AbstractMCMC ~/.julia/packages/AbstractMCMC/FSyVk/src/sample.jl:142
[21] sample(rng::TaskLocalRNG, model::DynamicPPL.Model{typeof(demo), (), (), (), Tuple{}, Tuple{}, DynamicPPL.DefaultContext}, sampler::DynamicPPL.Sampler{Turing.Inference.ExternalSampler{MALA{Int64, UniformScaling{Bool}}, AutoForwardDiff{nothing, Nothing}, false}}, N::Int64; chain_type::Type, resume_from::Nothing, initial_state::Nothing, kwargs::@Kwargs{progress::Bool})
@ DynamicPPL ~/.julia/packages/DynamicPPL/cvlfK/src/sampler.jl:107
[22] sample
@ ~/.julia/packages/DynamicPPL/cvlfK/src/sampler.jl:97 [inlined]
[23] #sample#4
@ ~/.julia/packages/Turing/NQDYt/src/mcmc/Inference.jl:303 [inlined]
[24] sample
@ ~/.julia/packages/Turing/NQDYt/src/mcmc/Inference.jl:294 [inlined]
[25] #sample#3
@ ~/.julia/packages/Turing/NQDYt/src/mcmc/Inference.jl:291 [inlined]
[26] top-level scope
@ ~/work/docs/docs/tutorials/docs-17-implementing-samplers/index.qmd:457
As expected, we run into a DomainError
at some point, while if we set unconstrained=true
, letting Turing.jl transform the model to a unconstrained form behind the scenes, everything works as expected:
sample(demo(), externalsampler(sampler; unconstrained=true), 10_000; progress=false)
Chains MCMC chain (10000×3×1 Array{Float64, 3}):
Iterations = 1:1:10000
Number of chains = 1
Samples per chain = 10000
Wall duration = 1.94 seconds
Compute duration = 1.94 seconds
parameters = σ², x
internals = lp
Summary Statistics
parameters mean std mcse ess_bulk ess_tail rhat e ⋯
Symbol Float64 Float64 Float64 Float64 Float64 Float64 ⋯
σ² 0.4388 0.0000 0.0000 NaN NaN NaN ⋯
x 3.6326 0.0000 0.0000 NaN NaN NaN ⋯
1 column omitted
Quantiles
parameters 2.5% 25.0% 50.0% 75.0% 97.5%
Symbol Float64 Float64 Float64 Float64 Float64
σ² 0.4388 0.4388 0.4388 0.4388 0.4388
x 3.6326 3.6326 3.6326 3.6326 3.6326
Neat!
Similarly, which automatic differentiation backend one should use can be specified through the adtype
keyword argument too. For example, if we want to use ReverseDiff.jl instead of the default ForwardDiff.jl:
using ReverseDiff: ReverseDiff
# Specify that we want to use `AutoReverseDiff`.
sample(
demo(),
externalsampler(sampler; unconstrained=true, adtype=AutoReverseDiff()),
10_000;
=false
progress )
Chains MCMC chain (10000×3×1 Array{Float64, 3}):
Iterations = 1:1:10000
Number of chains = 1
Samples per chain = 10000
Wall duration = 3.59 seconds
Compute duration = 3.59 seconds
parameters = σ², x
internals = lp
Summary Statistics
parameters mean std mcse ess_bulk ess_tail rhat e ⋯
Symbol Float64 Float64 Float64 Float64 Float64 Float64 ⋯
σ² 0.8547 0.5120 0.0168 752.9085 504.7980 1.0064 ⋯
x -0.0043 0.9168 0.0388 692.7003 412.9017 1.0036 ⋯
1 column omitted
Quantiles
parameters 2.5% 25.0% 50.0% 75.0% 97.5%
Symbol Float64 Float64 Float64 Float64 Float64
σ² 0.1550 0.4333 0.7724 1.1950 2.0023
x -2.1356 -0.4317 0.0211 0.4642 1.8576
Double-neat.
Summary
At this point it’s worth maybe reminding ourselves what we did and also why we did it:
- We define our models in the LogDensityProblems.jl interface because it makes the sampler agnostic to how the underlying model is implemented.
- We implement our sampler in the AbstractMCMC.jl interface, which just means that our sampler is a subtype of
AbstractMCMC.AbstractSampler
and we implement the MCMC transition inAbstractMCMC.step
. - Points 1 and 2 makes it so our sampler can be used with a wide range of model implementations, amongst them being models implemented in both Turing.jl and Stan. This gives you, the inference implementer, a large collection of models to test your inference method on, in addition to allowing users of Turing.jl and Stan to try out your inference method with minimal effort.
Footnotes
We’re going with the leapfrog formulation because in a future version of this tutorial we’ll add a section extending this simple “baseline” MALA sampler to more complex versions. See issue #479 for progress on this.↩︎
There is no such thing as a proper interface in Julia (at least not officially), and so we use the word “interface” here to mean a few minimal methods that needs to be implemented by any type that we treat as a target model.↩︎