defmodule Chi2fit.Distribution.Wald do
# Copyright 2019 Pieter Rijken
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
@moduledoc """
Wald or Inverse Gauss distribution.
"""
defstruct [:pars, name: "wald"]
@type t() :: %__MODULE__{
pars: [number()] | nil,
name: String.t
}
end
defimpl Chi2fit.Distribution, for: Chi2fit.Distribution.Wald do
alias Chi2fit.Distribution, as: D
import D.Wald
alias D.Wald
@spec phi(x :: float) :: float
defp phi(x) do
d = %D.Normal{pars: [0.0, 1.0]}
D.cdf(d).(x)
end
@spec wald(mu::number(),lambda::number()) :: ((...) -> number)
defp wald(mu,lambda) when is_number(mu) and is_number(lambda) do
d = %D.Normal{pars: [0.0,1.0]}
fn () ->
w = D.random(d)
y = w*w
x = mu + mu*mu*y/2/lambda - mu/2/lambda*:math.sqrt(4*mu*lambda*y + mu*mu*y*y)
z = :rand.uniform()
if z <= mu/(mu+x), do: x, else: mu*mu/x
end
end
defp wald([avg: average],lambda), do: wald(average,lambda)
@spec waldCDF(number,number) :: (number -> number)
defp waldCDF(mu,_) when mu < 0, do: raise ArithmeticError, "Wald is only defined for positive average"
defp waldCDF(_,lambda) when lambda < 0, do: raise ArithmeticError, "Wald is only defined for positive shape"
defp waldCDF(mu,lambda) do
fn
x when x == 0 -> 0.0
x when x < 0 -> 0.0
x when x > 0 ->
phi(:math.sqrt(lambda/x) * (x/mu-1.0)) + :math.exp(2.0*lambda/mu) * phi(-:math.sqrt(lambda/x) * (x/mu+1.0))
end
end
def skewness(%Wald{pars: nil}), do: fn [k,lambda] -> 3*:math.sqrt(k/lambda) end
def kurtosis(%Wald{pars: nil}), do: fn [k,lambda] -> 15*k/lambda end
def size(%Wald{}), do: 2
def cdf(%Wald{pars: nil}), do: fn x,[k,lambda] -> waldCDF(k,lambda).(x) end
def pdf(%Wald{pars: nil}), do: fn x, [mu,lambda] -> :math.sqrt(lambda/2/:math.pi/x/x/x) * :math.exp( -lambda*(x-mu)*(x-mu)/2/x/mu/mu ) end
def random(%Wald{pars: [k,lambda]}), do: wald(k,lambda).()
def random(%Wald{pars: nil}), do: fn [k,lambda] -> wald(k,lambda).() end
def name(model), do: model.name
end
defimpl Inspect, for: Chi2fit.Distribution.Wald do
import Inspect.Algebra
def inspect(dict, opts) do
case dict.pars do
nil ->
"#Wald<>"
[k,lambda] ->
concat ["#Wald<", to_doc("k=#{k}, lambda=#{lambda}", opts), ">"]
list ->
concat ["#Wald<", to_doc(list, opts), ">"]
end
end
end