-
Notifications
You must be signed in to change notification settings - Fork 3
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Type Parameters for ObservationProcess #59
Comments
I think you raise a very valid criticism of my idea. First of all, I agree that it doesn't make a whole lot of sense in terms of type parameterization. They both have a place, but I feel like we more commonly use the log-likelihood type for inference algorithms, so I focused on that element. Honestly, there is a world in which they coexist, but I think adding an additional parameter would just be extra clutter. In Distributions.jl, the return type of Most of this is to say that we can instead of using the type parameter to determine the logpdf, we can use some type promotion given |
I'm not sure I'm quite following what you're saying but I agree with the general idea that it is clear cluttered to add loads of type parameters to handle both the log-likelihood type and state types. My latest thoughts on this that the type parameters for the state/observation types are perhaps the overkill bit. Yes it's a bit more messy in the code, but the following is (I think—would appreciate confirmation) completely valid and type stable
or something like that. There would therefore only be one type parameter which would correspond to the log-likelihood type but more generally the floating point type that all fields in the model use (e.g. Q::Matrix{T}). If you're happy with this in terms of autodiff, I think that's likely sufficient and avoids over-complication. |
the above is not necessarily type stable. Consider the scanario in which the initialization is |
Ah I think I see what you mean. I guess I was suggesting this is the users' responsibility. The would have to define
so that the type of A is known and is guaranteed to match the initialisation. Is this what you mean? |
yeah, although it would have to be the other way around. The type of |
Exactly, though on reflection maybe that is a tough ask in general. I think I've shared with you before that Distributions isn't always type consistent. E.g. Uniform{T} always returns Float64 when sampling regardless of T. |
Well that sucks. I honestly might raise an issue. |
I checked just to confirm and oh my goodness 🤮 julia> test = Uniform(0f0,1f0)
Uniform{Float32}(a=0.0f0, b=1.0f0)
julia> rand(test)
0.07834174250700543
julia> test = Uniform(BigFloat(0),BigFloat(1))
Uniform{BigFloat}(a=0.0, b=1.0)
julia> rand(test)
0.68051738282173801497521026249160058796405792236328125
julia> rand(test, 2)
2-element Vector{Float64}:
0.8438465943304864
0.821132972814556 |
Ikr. There's actually a long history of issues on this topic. Think the latest is here: Think there has been some resistance to it for historical reasons, though some progress is being made. But most other distributions can't be relied on right now. |
I've been thinking of a possible third approach which whilst even more complicated, provides all the expressivity we need without the awkward The proposal would be to have two types for each of dyn and obs. The first would be the "arithmetic type" and be something like Float32, Float64, Dual etc. The second type would be the state/observation type—and it would be up to the user to make sure this matches the arithmetic type. We then obtain the SSM type by promoting the two arithmetic types (or forcing them to be the same—I think this makes more sense). With this approach, we can still preallocate memory for states/observations. Maybe overkill, but let me know what you think. |
Why do we need to parametrize the SSM with the loglikelihood type ? The states type is useful to dispatch / allocate but if we need the loglikelihood type, and it's not straightforward to derive from the state types (with the caveats around Distributions.jl instability ...), we could just have a helper function ? |
@charlesknipp might be better suited to answer this as recall it being his addition
I started the above change and found it led to a load of issues that would be faffy to fix. For example, when creating the container for the dense particle storage, you need to know the element type of the latent dynamics. You could find this by simulating x0 and then looking at its type but this seems like a unintuitive and error-prone workflow. I'm going to return the above suggestion of having one type for the arithmetic type and one for the state/observation element type. I will make this change now. Maybe the arithmetic type can be dropped and computed using a helper as Frederic suggests, but I will leave that for after the LAFI workshop. |
On the point of creating a helper to infer these types, would this depend on something like julia> struct LinearTransformation{T}
A::Matrix{T}
b::Vector{T}
end
julia> transform(t::LinearTransformation, x) = t.A * x + b
transform (generic function with 1 method)
julia> t = LinearTransformation(rand(2, 2), rand(2))
LinearTransformation{Float64}([0.25737465916134483 0.17756543003788494; 0.07369227300887338 0.597478835391764], [0.7104961946389236, 0.11151281420460368])
julia> x = rand(Float32, 2)
2-element Vector{Float32}:
0.79233974
0.34546995
julia> t.A * x + t.b
2-element Vector{Float64}:
0.9757678862497214
0.37631311516390975
julia> Base.return_types(transform, (LinearTransformation{Float64}, Vector{Float32}))
1-element Vector{Any}:
Any I'm not sure where |
Is it just a typo on |
The motivation for this was to easily be able to preallocate probability weights and initialize log-likelihoods without additional function calls. For a single model, this is all information we know at creation (ideally) and shouldn't require additional function calls. This is also useful for something like ForwardDiff, where the log-likelihoods should be initialized with
This is what we should do now, but I also worry that there's no easy way to determine what the likelihood type should be without running a computation every time we call this helper. I'm not sure how other modules handle cases like this, since I think Turing just operates with Float64s by default. |
Oh of course. I should have spotted that myself. Thanks. So in theory, that's the sort of thing our potentially helper would be doing. Not sure how expensive such a call is. |
It looks like at some point we've diverged on the interpretation of the type parameter for
ObservationProcess
.As a recap, we have just the one type parameter
This is used in two places with contradictory meanings. First it is used to determine the output container type for forward simulation:
SSMProblems.jl/src/utils/forward_simulation.jl
Lines 9 to 13 in 672da2f
It is also used to define the main type parameter of the SSM which I believe corresponds to the type of the log-likelihoods generated by the SSM:
SSMProblems.jl/src/SSMProblems.jl
Lines 258 to 259 in 672da2f
I feel like both of these use-cases have a place. Is it worth having two type parameters?
One for the element type (which could be some weird abstract thing, e.g. a collection of Lévy jumps, rather than just a Vector{T}). One for the type returned by the logdensity function?
If we are doing this, the same should probably hold for the LatentDynamics. One for the state type, one for the type returned by logdensity.
Alternatively, it might be overkill to store the type of the state/observation. Are there any serious drawbacks that come from sampling x1, y1 and then constructing a vector from their empirical types?
Would especially appreciate thoughts from @charlesknipp as I believe it was your desire to store the log-likelihood type.
The text was updated successfully, but these errors were encountered: