Skip to content

Commit 5311dff

Browse files
add beta binomial distribution (#234)
* add beta binomial distribution * run JuliaFormatter * Apply suggestions from code review Co-authored-by: Chad Scherrer <[email protected]> * remove proxy, bump version Co-authored-by: Chad Scherrer <[email protected]>
1 parent 6a42ef6 commit 5311dff

File tree

4 files changed

+49
-6
lines changed

4 files changed

+49
-6
lines changed

Project.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
name = "MeasureTheory"
22
uuid = "eadaa1a4-d27c-401d-8699-e962e1bbc33b"
33
authors = ["Chad Scherrer <[email protected]> and contributors"]
4-
version = "0.17.2"
4+
version = "0.17.3"
55

66
[deps]
77
Accessors = "7d9f7c33-5ae7-4f3b-8dc6-eff91059b697"

src/MeasureTheory.jl

+1
Original file line numberDiff line numberDiff line change
@@ -155,6 +155,7 @@ include("parameterized/binomial.jl")
155155
include("parameterized/multinomial.jl")
156156
include("parameterized/lkj-cholesky.jl")
157157
include("parameterized/negativebinomial.jl")
158+
include("parameterized/betabinomial.jl")
158159
include("parameterized/gamma.jl")
159160
include("parameterized/snedecorf.jl")
160161
include("parameterized/inverse-gaussian.jl")

src/parameterized/betabinomial.jl

+37
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
# Beta-Binomial distribution
2+
3+
export BetaBinomial
4+
import Base
5+
using SpecialFunctions
6+
7+
@parameterized BetaBinomial(n, α, β)
8+
9+
basemeasure(d::BetaBinomial) = CountingMeasure()
10+
11+
testvalue(::BetaBinomial) = 0
12+
13+
@kwstruct BetaBinomial(n, α, β)
14+
15+
function Base.rand(
16+
rng::AbstractRNG,
17+
::Type{T},
18+
d::BetaBinomial{(:n, :α, :β)},
19+
) where {T}
20+
k = rand(rng, T, Beta(d.α, d.β))
21+
return rand(rng, T, Binomial(d.n, k))
22+
end
23+
24+
@inline function insupport(d::BetaBinomial, x)
25+
isinteger(x) && 0 x d.n
26+
end
27+
28+
@inline function logdensity_def(d::BetaBinomial{(:n, :α, :β)}, y)
29+
(n, α, β) = (d.n, d.α, d.β)
30+
logbinom = -log1p(n) - logbeta(y + 1, n - y + 1)
31+
lognum = logbeta(y + α, n - y + β)
32+
logdenom = logbeta(α, β)
33+
return logbinom + lognum - logdenom
34+
end
35+
36+
asparams(::Type{<:BetaBinomial}, ::StaticSymbol{:α}) = asℝ₊
37+
asparams(::Type{<:BetaBinomial}, ::StaticSymbol{:β}) = asℝ₊

test/runtests.jl

+10-5
Original file line numberDiff line numberDiff line change
@@ -54,6 +54,7 @@ test_measures = Any[
5454
Bernoulli(0.2)
5555
Beta(2, 3)
5656
Binomial(10, 0.3)
57+
BetaBinomial(10, 2, 3)
5758
Cauchy()
5859
Dirichlet(ones(3))
5960
Exponential()
@@ -277,7 +278,7 @@ end
277278

278279
@testset "Product of Diracs" begin
279280
x = randn(3)
280-
t = as(productmeasure(Dirac.(x)))
281+
t = as(productmeasure(Dirac.(x)))
281282
@test transform(t, []) == x
282283
end
283284

@@ -297,7 +298,7 @@ end
297298

298299
# chain = Chain(kernel, μ)
299300

300-
# dyniterate(iter::TimeLift, ::Nothing) = dyniterate(iter, 0=>nothing)
301+
# dyniterate(iter::TimeLift, ::Nothing) = dyniterate(iter, 0=>nothing)
301302
# tr1 = trace(TimeLift(chain), nothing, u -> u[1] > 15)
302303
# tr2 = trace(TimeLift(rand(Random.GLOBAL_RNG, chain)), nothing, u -> u[1] > 15)
303304
# collect(Iterators.take(chain, 10))
@@ -348,8 +349,8 @@ end
348349
# NOTE: The `test_broken` below are mostly because of the change to `Affine`.
349350
# For example, `Normal{(:μ,:σ)}` is now `Affine{(:μ,:σ), Normal{()}}`.
350351
# The problem is not really with these measures, but with the tests
351-
# themselves.
352-
#
352+
# themselves.
353+
#
353354
# We should instead probably be doing e.g.
354355
# `D = typeof(Normal(μ=0.3, σ=4.1))`
355356

@@ -371,6 +372,10 @@ end
371372
@test repro(Beta, (, ))
372373
end
373374

375+
@testset "BetaBinomial" begin
376+
@test repro(BetaBinomial, (:n, , ), (n = 10,))
377+
end
378+
374379
@testset "Cauchy" begin
375380
@test_broken repro(Cauchy, (, ))
376381
end
@@ -652,7 +657,7 @@ end
652657
end
653658

654659
x = rand(d)
655-
660+
656661
@test logdensityof(d, x) isa Real
657662
end
658663

0 commit comments

Comments
 (0)