# Copyright (c) 2024 zfit
from __future__ import annotations
from typing import Literal
import numpy as np
import pydantic
import tensorflow as tf
import zfit.z.numpy as znp
from zfit import z
from ..core.basepdf import BasePDF
from ..core.serialmixin import SerializableMixin
from ..core.space import ANY_LOWER, ANY_UPPER, Space, supports
from ..serialization import Serializer, SpaceRepr
from ..serialization.pdfrepr import BasePDFRepr
from ..util import ztyping
from ..util.ztyping import ExtendedInputType, NormInputType
def _powerlaw(x, a, k):
return a * znp.power(x, k)
@z.function(wraps="tensor", keepalive=True)
def crystalball_func(x, mu, sigma, alpha, n):
t = (x - mu) / sigma * tf.sign(alpha)
abs_alpha = znp.abs(alpha)
a = znp.power((n / abs_alpha), n) * znp.exp(-0.5 * znp.square(alpha))
b = (n / abs_alpha) - abs_alpha
cond = tf.less(t, -abs_alpha)
func = z.safe_where(
cond,
lambda t: _powerlaw(b - t, a, -n),
lambda t: znp.exp(-0.5 * znp.square(t)),
values=t,
value_safer=lambda t: znp.ones_like(t) * (b - 2),
)
return znp.maximum(func, znp.zeros_like(func))
@z.function(wraps="tensor", keepalive=True, stateless_args=False)
def double_crystalball_func(x, mu, sigma, alphal, nl, alphar, nr):
cond = tf.less(x, mu)
return tf.where(
cond,
crystalball_func(x, mu, sigma, alphal, nl),
crystalball_func(x, mu, sigma, -alphar, nr),
)
@z.function(wraps="tensor", keepalive=True, stateless_args=False)
def generalized_crystalball_func(x, mu, sigmal, alphal, nl, sigmar, alphar, nr):
cond = tf.less(x, mu)
return tf.where(
cond,
crystalball_func(x, mu, sigmal, alphal, nl),
crystalball_func(x, mu, sigmar, -alphar, nr),
)
# created with the help of TensorFlow autograph used on python code converted from ShapeCB of RooFit
def crystalball_integral(limits, params, model):
del model
mu = params["mu"]
sigma = params["sigma"]
alpha = params["alpha"]
n = params["n"]
lower, upper = limits._rect_limits_tf
return crystalball_integral_func(mu, sigma, alpha, n, lower, upper)
@z.function(wraps="tensor", keepalive=True)
# @tf.function # BUG? TODO: problem with tf.function and input signature
def crystalball_integral_func(mu, sigma, alpha, n, lower, upper):
sqrt_pi_over_two = np.sqrt(np.pi / 2)
sqrt2 = np.sqrt(2)
use_log = tf.less(znp.abs(n - 1.0), 1e-05)
abs_sigma = znp.abs(sigma)
abs_alpha = znp.abs(alpha)
tmin = (lower - mu) / abs_sigma
tmax = (upper - mu) / abs_sigma
alpha_negative = tf.less(alpha, 0)
# do not move on two lines, logic will fail...
tmax, tmin = (
znp.where(alpha_negative, -tmin, tmax),
znp.where(alpha_negative, -tmax, tmin),
)
if_true_4 = abs_sigma * sqrt_pi_over_two * (tf.math.erf(tmax / sqrt2) - tf.math.erf(tmin / sqrt2))
a = znp.power(n / abs_alpha, n) * znp.exp(-0.5 * tf.square(abs_alpha))
b = n / abs_alpha - abs_alpha
# gradients from tf.where can be NaN if the non-selected branch is NaN
# https://github.com/tensorflow/tensorflow/issues/42889
# solution is to provide save values for the non-selected branch to never make them become NaNs
b_tmin = b - tmin
safe_b_tmin_ones = znp.where(b_tmin > 0, b_tmin, znp.ones_like(b_tmin))
b_tmax = b - tmax
safe_b_tmax_ones = znp.where(b_tmax > 0, b_tmax, znp.ones_like(b_tmax))
if_true_1 = a * abs_sigma * (znp.log(safe_b_tmin_ones) - znp.log(safe_b_tmax_ones))
if_false_1 = (
a
* abs_sigma
/ (1.0 - n)
* (1.0 / znp.power(safe_b_tmin_ones, n - 1.0) - 1.0 / znp.power(safe_b_tmax_ones, n - 1.0))
)
if_true_3 = tf.where(use_log, if_true_1, if_false_1)
if_true_2 = a * abs_sigma * (znp.log(safe_b_tmin_ones) - znp.log(n / abs_alpha))
if_false_2 = (
a
* abs_sigma
/ (1.0 - n)
* (1.0 / znp.power(safe_b_tmin_ones, n - 1.0) - 1.0 / znp.power(n / abs_alpha, n - 1.0))
)
term1 = tf.where(use_log, if_true_2, if_false_2)
term2 = abs_sigma * sqrt_pi_over_two * (tf.math.erf(tmax / sqrt2) - tf.math.erf(-abs_alpha / sqrt2))
if_false_3 = term1 + term2
if_false_4 = tf.where(tf.less_equal(tmax, -abs_alpha), if_true_3, if_false_3)
# if_false_4()
result = tf.where(tf.greater_equal(tmin, -abs_alpha), if_true_4, if_false_4)
if result.shape.rank != 0:
result = tf.gather(result, 0, axis=-1) # remove last dim, should vanish
return result
def double_crystalball_mu_integral(limits, params, model):
del model
mu = params["mu"]
sigma = params["sigma"]
alphal = params["alphal"]
nl = params["nl"]
alphar = params["alphar"]
nr = params["nr"]
lower, upper = limits._rect_limits_tf
lower = lower[:, 0]
upper = upper[:, 0]
return double_crystalball_mu_integral_func(
mu=mu,
sigma=sigma,
alphal=alphal,
nl=nl,
alphar=alphar,
nr=nr,
lower=lower,
upper=upper,
)
# @z.function(wraps="tensor") # TODO: this errors, fro whatever reason?
def double_crystalball_mu_integral_func(mu, sigma, alphal, nl, alphar, nr, lower, upper):
# mu_broadcast =
upper_of_lowerint = znp.minimum(mu, upper)
integral_left = crystalball_integral_func(
mu=mu, sigma=sigma, alpha=alphal, n=nl, lower=lower, upper=upper_of_lowerint
)
left = tf.where(tf.less(mu, lower), znp.zeros_like(integral_left), integral_left)
lower_of_upperint = znp.maximum(mu, lower)
integral_right = crystalball_integral_func(
mu=mu, sigma=sigma, alpha=-alphar, n=nr, lower=lower_of_upperint, upper=upper
)
right = tf.where(tf.greater(mu, upper), znp.zeros_like(integral_right), integral_right)
return left + right
def generalized_crystalball_mu_integral(limits, params, model):
del model
mu = params["mu"]
sigmal = params["sigmal"]
alphal = params["alphal"]
nl = params["nl"]
sigmar = params["sigmar"]
alphar = params["alphar"]
nr = params["nr"]
lower, upper = limits._rect_limits_tf
lower = lower[:, 0]
upper = upper[:, 0]
return generalized_crystalball_mu_integral_func(
mu=mu,
sigmal=sigmal,
alphal=alphal,
nl=nl,
sigmar=sigmar,
alphar=alphar,
nr=nr,
lower=lower,
upper=upper,
)
# @z.function(wraps="tensor") # TODO: this errors, for whatever reason, when running example/signal_bkg_mass_extended_fit_binned.py
def generalized_crystalball_mu_integral_func(mu, sigmal, alphal, nl, sigmar, alphar, nr, lower, upper):
# mu_broadcast =
upper_of_lowerint = znp.minimum(mu, upper)
integral_left = crystalball_integral_func(
mu=mu, sigma=sigmal, alpha=alphal, n=nl, lower=lower, upper=upper_of_lowerint
)
left = tf.where(tf.less(mu, lower), znp.zeros_like(integral_left), integral_left)
lower_of_upperint = znp.maximum(mu, lower)
integral_right = crystalball_integral_func(
mu=mu, sigma=sigmar, alpha=-alphar, n=nr, lower=lower_of_upperint, upper=upper
)
right = tf.where(tf.greater(mu, upper), znp.zeros_like(integral_right), integral_right)
return left + right
[docs]
class CrystalBall(BasePDF, SerializableMixin):
_N_OBS = 1
def __init__(
self,
mu: ztyping.ParamTypeInput,
sigma: ztyping.ParamTypeInput,
alpha: ztyping.ParamTypeInput,
n: ztyping.ParamTypeInput,
obs: ztyping.ObsTypeInput,
*,
extended: ExtendedInputType = None,
norm: NormInputType = None,
name: str = "CrystalBall",
label: str | None = None,
):
"""Crystal Ball shaped PDF. A combination of a Gaussian with a powerlaw tail.
The function is defined as follows:
.. math::
f(x;\\mu, \\sigma, \\alpha, n) = \\begin{cases} \\exp(- \\frac{(x - \\mu)^2}{2 \\sigma^2}),
& \\mbox{for }\\frac{x - \\mu}{\\sigma} \\geqslant -\\alpha \\newline
A \\cdot (B - \\frac{x - \\mu}{\\sigma})^{-n}, & \\mbox{for }\\frac{x - \\mu}{\\sigma}
< -\\alpha \\end{cases}
with
.. math::
A = \\left(\\frac{n}{\\left| \\alpha \\right|}\\right)^n \\cdot
\\exp\\left(- \\frac {\\left|\\alpha \\right|^2}{2}\\right)
B = \\frac{n}{\\left| \\alpha \\right|} - \\left| \\alpha \\right|
Args:
mu: The mean of the gaussian
sigma: Standard deviation of the gaussian
alpha: parameter where to switch from a gaussian to the powertail
n: Exponent of the powertail
obs: |@doc:pdf.init.obs| Observables of the
model. This will be used as the default space of the PDF and,
if not given explicitly, as the normalization range.
The default space is used for example in the sample method: if no
sampling limits are given, the default space is used.
If the observables are binned and the model is unbinned, the
model will be a binned model, by wrapping the model in a
:py:class:`~zfit.pdf.BinnedFromUnbinnedPDF`, equivalent to
calling :py:meth:`~zfit.pdf.BasePDF.to_binned`.
The observables are not equal to the domain as it does not restrict or
truncate the model outside this range. |@docend:pdf.init.obs|
extended: |@doc:pdf.init.extended| The overall yield of the PDF.
If this is parameter-like, it will be used as the yield,
the expected number of events, and the PDF will be extended.
An extended PDF has additional functionality, such as the
``ext_*`` methods and the ``counts`` (for binned PDFs). |@docend:pdf.init.extended|
norm: |@doc:pdf.init.norm| Normalization of the PDF.
By default, this is the same as the default space of the PDF. |@docend:pdf.init.norm|
name: |@doc:pdf.init.name| Name of the PDF.
Maybe has implications on the serialization and deserialization of the PDF.
For a human-readable name, use the label. |@docend:pdf.init.name|
label: |@doc:pdf.init.label| Human-readable name
or label of
the PDF for a better description, to be used with plots etc.
Has no programmatical functional purpose as identification. |@docend:pdf.init.label|
.. _CBShape: https://en.wikipedia.org/wiki/Crystal_Ball_function
"""
params = {"mu": mu, "sigma": sigma, "alpha": alpha, "n": n}
super().__init__(obs=obs, name=name, params=params, extended=extended, norm=norm, label=label)
@supports(norm=False)
def _pdf(self, x, norm, params):
del norm
mu = params["mu"]
sigma = params["sigma"]
alpha = params["alpha"]
n = params["n"]
x = z.unstack_x(x)
return crystalball_func(x=x, mu=mu, sigma=sigma, alpha=alpha, n=n)
class CrystalBallPDFRepr(BasePDFRepr):
_implementation = CrystalBall
hs3_type: Literal["CrystalBall"] = pydantic.Field("CrystalBall", alias="type")
x: SpaceRepr
mu: Serializer.types.ParamTypeDiscriminated
sigma: Serializer.types.ParamTypeDiscriminated
alpha: Serializer.types.ParamTypeDiscriminated
n: Serializer.types.ParamTypeDiscriminated
crystalball_integral_limits = Space(axes=(0,), lower=ANY_LOWER, upper=ANY_UPPER)
CrystalBall.register_analytic_integral(func=crystalball_integral, limits=crystalball_integral_limits)
[docs]
class DoubleCB(BasePDF, SerializableMixin):
_N_OBS = 1
def __init__(
self,
mu: ztyping.ParamTypeInput,
sigma: ztyping.ParamTypeInput,
alphal: ztyping.ParamTypeInput,
nl: ztyping.ParamTypeInput,
alphar: ztyping.ParamTypeInput,
nr: ztyping.ParamTypeInput,
obs: ztyping.ObsTypeInput,
*,
extended: ExtendedInputType = None,
norm: NormInputType = None,
name: str = "DoubleCB",
label: str | None = None,
):
"""Double-sided Crystal Ball shaped PDF. A combination of two CB using the **mu** (not a frac) on each side.
The function is defined as follows:
.. math::
f(x;\\mu, \\sigma, \\alpha_{L}, n_{L}, \\alpha_{R}, n_{R}) = \\begin{cases}
A_{L} \\cdot (B_{L} - \\frac{x - \\mu}{\\sigma})^{-n_{L}},
& \\mbox{for }\\frac{x - \\mu}{\\sigma} < -\\alpha_{L} \\newline
\\exp(- \\frac{(x - \\mu)^2}{2 \\sigma^2}),
& \\mbox{for }-\\alpha_{L} \\leqslant \\frac{x - \\mu}{\\sigma} \\leqslant \\alpha_{R} \\newline
A_{R} \\cdot (B_{R} + \\frac{x - \\mu}{\\sigma})^{-n_{R}},
& \\mbox{for }\\frac{x - \\mu}{\\sigma} > \\alpha_{R}
\\end{cases}
with
.. math::
A_{L/R} = \\left(\\frac{n_{L/R}}{\\left| \\alpha_{L/R} \\right|}\\right)^n_{L/R} \\cdot
\\exp\\left(- \\frac {\\left|\\alpha_{L/R} \\right|^2}{2}\\right)
B_{L/R} = \\frac{n_{L/R}}{\\left| \\alpha_{L/R} \\right|} - \\left| \\alpha_{L/R} \\right|
Args:
mu: The mean of the gaussian
sigma: Standard deviation of the gaussian
alphal: parameter where to switch from a gaussian to the powertail on the left
side
nl: Exponent of the powertail on the left side
alphar: parameter where to switch from a gaussian to the powertail on the right
side
nr: Exponent of the powertail on the right side
obs: |@doc:pdf.init.obs| Observables of the
model. This will be used as the default space of the PDF and,
if not given explicitly, as the normalization range.
The default space is used for example in the sample method: if no
sampling limits are given, the default space is used.
If the observables are binned and the model is unbinned, the
model will be a binned model, by wrapping the model in a
:py:class:`~zfit.pdf.BinnedFromUnbinnedPDF`, equivalent to
calling :py:meth:`~zfit.pdf.BasePDF.to_binned`.
The observables are not equal to the domain as it does not restrict or
truncate the model outside this range. |@docend:pdf.init.obs|
extended: |@doc:pdf.init.extended| The overall yield of the PDF.
If this is parameter-like, it will be used as the yield,
the expected number of events, and the PDF will be extended.
An extended PDF has additional functionality, such as the
``ext_*`` methods and the ``counts`` (for binned PDFs). |@docend:pdf.init.extended|
norm: |@doc:pdf.init.norm| Normalization of the PDF.
By default, this is the same as the default space of the PDF. |@docend:pdf.init.norm|
name: |@doc:pdf.init.name| Name of the PDF.
Maybe has implications on the serialization and deserialization of the PDF.
For a human-readable name, use the label. |@docend:pdf.init.name|
label: |@doc:pdf.init.label| Human-readable name
or label of
the PDF for a better description, to be used with plots etc.
Has no programmatical functional purpose as identification. |@docend:pdf.init.label|
"""
params = {
"mu": mu,
"sigma": sigma,
"alphal": alphal,
"nl": nl,
"alphar": alphar,
"nr": nr,
}
super().__init__(obs=obs, name=name, params=params, extended=extended, norm=norm, label=label)
def _unnormalized_pdf(self, x):
mu = self.params["mu"].value()
sigma = self.params["sigma"].value()
alphal = self.params["alphal"].value()
nl = self.params["nl"].value()
alphar = self.params["alphar"].value()
nr = self.params["nr"].value()
x = x.unstack_x()
return double_crystalball_func(
x=x,
mu=mu,
sigma=sigma,
alphal=alphal,
nl=nl,
alphar=alphar,
nr=nr,
)
class DoubleCBPDFRepr(BasePDFRepr):
_implementation = DoubleCB
hs3_type: Literal["DoubleCB"] = pydantic.Field("DoubleCB", alias="type")
x: SpaceRepr
mu: Serializer.types.ParamTypeDiscriminated
sigma: Serializer.types.ParamTypeDiscriminated
alphal: Serializer.types.ParamTypeDiscriminated
nl: Serializer.types.ParamTypeDiscriminated
alphar: Serializer.types.ParamTypeDiscriminated
nr: Serializer.types.ParamTypeDiscriminated
DoubleCB.register_analytic_integral(func=double_crystalball_mu_integral, limits=crystalball_integral_limits)
[docs]
class GeneralizedCB(BasePDF, SerializableMixin):
_N_OBS = 1
def __init__(
self,
mu: ztyping.ParamTypeInput,
sigmal: ztyping.ParamTypeInput,
alphal: ztyping.ParamTypeInput,
nl: ztyping.ParamTypeInput,
sigmar: ztyping.ParamTypeInput,
alphar: ztyping.ParamTypeInput,
nr: ztyping.ParamTypeInput,
obs: ztyping.ObsTypeInput,
*,
extended: ExtendedInputType = None,
norm: NormInputType = None,
name: str = "GeneralizedCB",
label: str | None = None,
):
"""Generalized asymmetric double-sided Crystal Ball shaped PDF. A combination of two CB using the **mu** (not a
frac) and a different **sigma** on each side.
The function is defined as follows:
.. math::
f(x;\\mu, \\sigma_{L}, \\alpha_{L}, n_{L}, \\sigma_{R}, \\alpha_{R}, n_{R}) = \\begin{cases}
A_{L} \\cdot (B_{L} - \\frac{x - \\mu}{\\sigma_{L}})^{-n_{L}},
& \\mbox{for }\\frac{x - \\mu}{\\sigma_{L}} < -\\alpha_{L} \\newline
\\exp(- \\frac{(x - \\mu)^2}{2 \\sigma_{L}^2}),
& \\mbox{for }-\\alpha_{L} \\leqslant \\frac{x - \\mu}{\\sigma_{L}} \\leqslant 0 \\newline
\\exp(- \\frac{(x - \\mu)^2}{2 \\sigma_{R}^2}),
& \\mbox{for }0 \\leqslant \\frac{x - \\mu}{\\sigma_{R}} \\leqslant \\alpha_{R} \\newline
A_{R} \\cdot (B_{R} + \\frac{x - \\mu}{\\sigma_{R}})^{-n_{R}},
& \\mbox{for }\\frac{x - \\mu}{\\sigma_{R}} > \\alpha_{R}
\\end{cases}
with
.. math::
A_{L/R} = \\left(\\frac{n_{L/R}}{\\left| \\alpha_{L/R} \\right|}\\right)^n_{L/R} \\cdot
\\exp\\left(- \\frac {\\left|\\alpha_{L/R} \\right|^2}{2}\\right)
B_{L/R} = \\frac{n_{L/R}}{\\left| \\alpha_{L/R} \\right|} - \\left| \\alpha_{L/R} \\right|
Args:
mu: The mean of the gaussian
sigmal: Standard deviation of the gaussian on the left side
alphal: parameter where to switch from a gaussian to the powertail on the left
side
nl: Exponent of the powertail on the left side
sigmar: Standard deviation of the gaussian on the right side
alphar: parameter where to switch from a gaussian to the powertail on the right
side
nr: Exponent of the powertail on the right side
obs: |@doc:pdf.init.obs| Observables of the
model. This will be used as the default space of the PDF and,
if not given explicitly, as the normalization range.
The default space is used for example in the sample method: if no
sampling limits are given, the default space is used.
If the observables are binned and the model is unbinned, the
model will be a binned model, by wrapping the model in a
:py:class:`~zfit.pdf.BinnedFromUnbinnedPDF`, equivalent to
calling :py:meth:`~zfit.pdf.BasePDF.to_binned`.
The observables are not equal to the domain as it does not restrict or
truncate the model outside this range. |@docend:pdf.init.obs|
extended: |@doc:pdf.init.extended| The overall yield of the PDF.
If this is parameter-like, it will be used as the yield,
the expected number of events, and the PDF will be extended.
An extended PDF has additional functionality, such as the
``ext_*`` methods and the ``counts`` (for binned PDFs). |@docend:pdf.init.extended|
norm: |@doc:pdf.init.norm| Normalization of the PDF.
By default, this is the same as the default space of the PDF. |@docend:pdf.init.norm|
name: |@doc:pdf.init.name| Name of the PDF.
Maybe has implications on the serialization and deserialization of the PDF.
For a human-readable name, use the label. |@docend:pdf.init.name|
label: |@doc:pdf.init.label| Human-readable name
or label of
the PDF for a better description, to be used with plots etc.
Has no programmatical functional purpose as identification. |@docend:pdf.init.label|
"""
params = {
"mu": mu,
"sigmal": sigmal,
"alphal": alphal,
"nl": nl,
"sigmar": sigmar,
"alphar": alphar,
"nr": nr,
}
super().__init__(obs=obs, name=name, params=params, extended=extended, norm=norm, label=label)
def _unnormalized_pdf(self, x):
mu = self.params["mu"].value()
sigmal = self.params["sigmal"].value()
alphal = self.params["alphal"].value()
sigmar = self.params["sigmar"].value()
nl = self.params["nl"].value()
alphar = self.params["alphar"].value()
nr = self.params["nr"].value()
x = x.unstack_x()
return generalized_crystalball_func(
x=x,
mu=mu,
sigmal=sigmal,
alphal=alphal,
sigmar=sigmar,
nl=nl,
alphar=alphar,
nr=nr,
)
class GeneralizedCBPDFRepr(BasePDFRepr):
_implementation = GeneralizedCB
hs3_type: Literal["GeneralizedCB"] = pydantic.Field("GeneralizedCB", alias="type")
x: SpaceRepr
mu: Serializer.types.ParamTypeDiscriminated
sigmal: Serializer.types.ParamTypeDiscriminated
alphal: Serializer.types.ParamTypeDiscriminated
sigmar: Serializer.types.ParamTypeDiscriminated
nl: Serializer.types.ParamTypeDiscriminated
alphar: Serializer.types.ParamTypeDiscriminated
nr: Serializer.types.ParamTypeDiscriminated
GeneralizedCB.register_analytic_integral(func=generalized_crystalball_mu_integral, limits=crystalball_integral_limits)
@z.function(wraps="tensor", keepalive=True)
def gaussexptail_func(x, mu, sigma, alpha):
t = (x - mu) / sigma * tf.sign(alpha)
abs_alpha = znp.abs(alpha)
cond = tf.less(t, -abs_alpha)
func = z.safe_where(
cond,
lambda t: 0.5 * znp.square(abs_alpha) + abs_alpha * t,
lambda t: -0.5 * znp.square(t),
values=t,
value_safer=lambda t: znp.ones_like(t) * abs_alpha,
)
return znp.exp(func)
@z.function(wraps="tensor", keepalive=True, stateless_args=False)
def generalized_gaussexptail_func(x, mu, sigmal, alphal, sigmar, alphar):
cond = tf.less(x, mu)
return tf.where(
cond,
gaussexptail_func(x, mu, sigmal, alphal),
gaussexptail_func(x, mu, sigmar, -alphar),
)
def gaussexptail_integral(limits, params, model):
del model
mu = params["mu"]
sigma = params["sigma"]
alpha = params["alpha"]
lower, upper = limits._rect_limits_tf
return gaussexptail_integral_func(mu, sigma, alpha, lower, upper)
@z.function(wraps="tensor", keepalive=True)
def gaussexptail_integral_func(mu, sigma, alpha, lower, upper):
sqrt_pi_over_two = np.sqrt(np.pi / 2)
sqrt2 = np.sqrt(2)
abs_sigma = znp.abs(sigma)
abs_alpha = znp.abs(alpha)
tmin = (lower - mu) / abs_sigma
tmax = (upper - mu) / abs_sigma
alpha_negative = tf.less(alpha, 0)
# do not move on two lines, logic will fail...
tmax, tmin = (
znp.where(alpha_negative, -tmin, tmax),
znp.where(alpha_negative, -tmax, tmin),
)
gauss_tmin_tmax_integral = abs_sigma * sqrt_pi_over_two * (tf.math.erf(tmax / sqrt2) - tf.math.erf(tmin / sqrt2))
exp_tmin_tmax_integral = (
abs_sigma
/ abs_alpha
* znp.exp(0.5 * znp.square(abs_alpha))
* (znp.exp(abs_alpha * tmax) - znp.exp(abs_alpha * tmin))
)
gauss_minus_abs_alpha_tmax_integral = (
abs_sigma * sqrt_pi_over_two * (tf.math.erf(tmax / sqrt2) - tf.math.erf(-abs_alpha / sqrt2))
)
exp_tmin_minus_abs_alpha_integral = (
abs_sigma
/ abs_alpha
* znp.exp(0.5 * znp.square(abs_alpha))
* (znp.exp(-znp.square(abs_alpha)) - znp.exp(abs_alpha * tmin))
)
integral_sum = exp_tmin_minus_abs_alpha_integral + gauss_minus_abs_alpha_tmax_integral
conditional_integral = tf.where(tf.less_equal(tmax, -abs_alpha), exp_tmin_tmax_integral, integral_sum)
result = tf.where(tf.greater_equal(tmin, -abs_alpha), gauss_tmin_tmax_integral, conditional_integral)
if result.shape.rank != 0:
result = tf.gather(result, 0, axis=-1)
return result
def generalized_gaussexptail_integral(limits, params, model):
del model
mu = params["mu"]
sigmal = params["sigmal"]
alphal = params["alphal"]
sigmar = params["sigmar"]
alphar = params["alphar"]
lower, upper = limits._rect_limits_tf
lower = lower[:, 0]
upper = upper[:, 0]
return generalized_gaussexptail_integral_func(
mu=mu,
sigmal=sigmal,
alphal=alphal,
sigmar=sigmar,
alphar=alphar,
lower=lower,
upper=upper,
)
@z.function(wraps="tensor", keepalive=True)
def generalized_gaussexptail_integral_func(mu, sigmal, alphal, sigmar, alphar, lower, upper):
upper_of_lowerint = znp.minimum(mu, upper)
integral_left = gaussexptail_integral_func(mu=mu, sigma=sigmal, alpha=alphal, lower=lower, upper=upper_of_lowerint)
left = tf.where(tf.less(mu, lower), znp.zeros_like(integral_left), integral_left)
lower_of_upperint = znp.maximum(mu, lower)
integral_right = gaussexptail_integral_func(
mu=mu, sigma=sigmar, alpha=-alphar, lower=lower_of_upperint, upper=upper
)
right = tf.where(tf.greater(mu, upper), znp.zeros_like(integral_right), integral_right)
return left + right
[docs]
class GaussExpTail(BasePDF, SerializableMixin):
_N_OBS = 1
def __init__(
self,
mu: ztyping.ParamTypeInput,
sigma: ztyping.ParamTypeInput,
alpha: ztyping.ParamTypeInput,
obs: ztyping.ObsTypeInput,
*,
extended: ExtendedInputType = None,
norm: NormInputType = None,
name: str = "GaussExpTail",
label: str | None = None,
):
"""GaussExpTail shaped PDF. A combination of a Gaussian with an exponential tail on one side.
The function is defined as follows:
.. math::
f(x;\\mu, \\sigma, \\alpha) = \\begin{cases} \\exp(- \\frac{(x - \\mu)^2}{2 \\sigma^2}),
& \\mbox{for }\\frac{x - \\mu}{\\sigma} \\geqslant -\\alpha \\newline
\\exp{\\left(\\frac{|\\alpha|^2}{2} + |\\alpha| \\left(\\frac{x - \\mu}{\\sigma}\\right)\\right)},
& \\mbox{for }\\frac{x - \\mu}{\\sigma} < -\\alpha \\end{cases}
Args:
mu: The mean of the gaussian
sigma: Standard deviation of the gaussian
alpha: parameter where to switch from a gaussian to the expontial tail
obs: |@doc:pdf.init.obs| Observables of the
model. This will be used as the default space of the PDF and,
if not given explicitly, as the normalization range.
The default space is used for example in the sample method: if no
sampling limits are given, the default space is used.
If the observables are binned and the model is unbinned, the
model will be a binned model, by wrapping the model in a
:py:class:`~zfit.pdf.BinnedFromUnbinnedPDF`, equivalent to
calling :py:meth:`~zfit.pdf.BasePDF.to_binned`.
The observables are not equal to the domain as it does not restrict or
truncate the model outside this range. |@docend:pdf.init.obs|
extended: |@doc:pdf.init.extended| The overall yield of the PDF.
If this is parameter-like, it will be used as the yield,
the expected number of events, and the PDF will be extended.
An extended PDF has additional functionality, such as the
``ext_*`` methods and the ``counts`` (for binned PDFs). |@docend:pdf.init.extended|
norm: |@doc:pdf.init.norm| Normalization of the PDF.
By default, this is the same as the default space of the PDF. |@docend:pdf.init.norm|
name: |@doc:pdf.init.name| Name of the PDF.
Maybe has implications on the serialization and deserialization of the PDF.
For a human-readable name, use the label. |@docend:pdf.init.name|
label: |@doc:pdf.init.label| Human-readable name
or label of
the PDF for a better description, to be used with plots etc.
Has no programmatical functional purpose as identification. |@docend:pdf.init.label|
"""
params = {"mu": mu, "sigma": sigma, "alpha": alpha}
super().__init__(obs=obs, name=name, params=params, extended=extended, norm=norm, label=label)
def _unnormalized_pdf(self, x):
mu = self.params["mu"].value()
sigma = self.params["sigma"].value()
alpha = self.params["alpha"].value()
x = z.unstack_x(x)
return gaussexptail_func(x=x, mu=mu, sigma=sigma, alpha=alpha)
class GaussExpTailPDFRepr(BasePDFRepr):
_implementation = GaussExpTail
hs3_type: Literal["GaussExpTail"] = pydantic.Field("GaussExpTail", alias="type")
x: SpaceRepr
mu: Serializer.types.ParamTypeDiscriminated
sigma: Serializer.types.ParamTypeDiscriminated
alpha: Serializer.types.ParamTypeDiscriminated
gaussexptail_integral_limits = Space(axes=0, limits=(ANY_LOWER, ANY_UPPER))
GaussExpTail.register_analytic_integral(func=gaussexptail_integral, limits=gaussexptail_integral_limits)
[docs]
class GeneralizedGaussExpTail(BasePDF, SerializableMixin):
_N_OBS = 1
def __init__(
self,
mu: ztyping.ParamTypeInput,
sigmal: ztyping.ParamTypeInput,
alphal: ztyping.ParamTypeInput,
sigmar: ztyping.ParamTypeInput,
alphar: ztyping.ParamTypeInput,
obs: ztyping.ObsTypeInput,
*,
extended: ExtendedInputType = None,
norm: NormInputType = None,
name: str = "GeneralizedGaussExpTail",
label: str | None = None,
):
"""GeneralizedGaussedExpTail shaped PDF which is Generalized assymetric double-sided GaussExpTail shaped PDF. A
combination of two GaussExpTail using the **mu** (not a frac) and a different **sigma** on each side.
The function is defined as follows:
.. math::
f(x;\\mu, \\sigma_{L}, \\alpha_{L}, \\sigma_{R}, \\alpha_{R}) = \\begin{cases}
\\exp{\\left(\\frac{|\\alpha_{L}|^2}{2} + |\\alpha_{L}| \\left(\\frac{x - \\mu}{\\sigma_{L}}\\right)\\right)},
& \\mbox{for }\\frac{x - \\mu}{\\sigma_{L}} < -\\alpha_{L} \\newline
\\exp(- \\frac{(x - \\mu)^2}{2 \\sigma_{L}^2}),
& \\mbox{for }-\\alpha_{L} \\leqslant \\frac{x - \\mu}{\\sigma_{L}} \\leqslant 0 \\newline
\\exp(- \\frac{(x - \\mu)^2}{2 \\sigma_{R}^2}),
& \\mbox{for }0 \\leqslant \\frac{x - \\mu}{\\sigma_{R}} \\leqslant \\alpha_{R} \\newline
\\exp{\\left(\\frac{|\\alpha_{R}|^2}{2} - |\\alpha_{R}| \\left(\\frac{x - \\mu}{\\sigma_{R}}\\right)\\right)},
& \\mbox{for }\\frac{x - \\mu}{\\sigma_{R}} > \\alpha_{R}
\\end{cases}
Args:
mu: The mean of the gaussian
sigmal: Standard deviation of the gaussian on the left side
alphal: parameter where to switch from a gaussian to the expontial tail on the left side
sigmar: Standard deviation of the gaussian on the right side
alphar: parameter where to switch from a gaussian to the expontial tail on the right side
obs: |@doc:pdf.init.obs| Observables of the
model. This will be used as the default space of the PDF and,
if not given explicitly, as the normalization range.
The default space is used for example in the sample method: if no
sampling limits are given, the default space is used.
If the observables are binned and the model is unbinned, the
model will be a binned model, by wrapping the model in a
:py:class:`~zfit.pdf.BinnedFromUnbinnedPDF`, equivalent to
calling :py:meth:`~zfit.pdf.BasePDF.to_binned`.
The observables are not equal to the domain as it does not restrict or
truncate the model outside this range. |@docend:pdf.init.obs|
extended: |@doc:pdf.init.extended| The overall yield of the PDF.
If this is parameter-like, it will be used as the yield,
the expected number of events, and the PDF will be extended.
An extended PDF has additional functionality, such as the
``ext_*`` methods and the ``counts`` (for binned PDFs). |@docend:pdf.init.extended|
norm: |@doc:pdf.init.norm| Normalization of the PDF.
By default, this is the same as the default space of the PDF. |@docend:pdf.init.norm|
name: |@doc:pdf.init.name| Name of the PDF.
Maybe has implications on the serialization and deserialization of the PDF.
For a human-readable name, use the label. |@docend:pdf.init.name|
label: |@doc:pdf.init.label| Human-readable name
or label of
the PDF for a better description, to be used with plots etc.
Has no programmatical functional purpose as identification. |@docend:pdf.init.label|
"""
params = {
"mu": mu,
"sigmal": sigmal,
"alphal": alphal,
"sigmar": sigmar,
"alphar": alphar,
}
super().__init__(obs=obs, name=name, params=params, extended=extended, norm=norm, label=label)
def _unnormalized_pdf(self, x):
mu = self.params["mu"].value()
sigmal = self.params["sigmal"].value()
alphal = self.params["alphal"].value()
sigmar = self.params["sigmar"].value()
alphar = self.params["alphar"].value()
x = z.unstack_x(x)
return generalized_gaussexptail_func(
x=x,
mu=mu,
sigmal=sigmal,
alphal=alphal,
sigmar=sigmar,
alphar=alphar,
)
class GeneralizedGaussExpTailPDFRepr(BasePDFRepr):
_implementation = GeneralizedGaussExpTail
hs3_type: Literal["GeneralizedGaussExpTail"] = pydantic.Field("GeneralizedGaussExpTail", alias="type")
x: SpaceRepr
mu: Serializer.types.ParamTypeDiscriminated
sigmal: Serializer.types.ParamTypeDiscriminated
alphal: Serializer.types.ParamTypeDiscriminated
sigmar: Serializer.types.ParamTypeDiscriminated
alphar: Serializer.types.ParamTypeDiscriminated
GeneralizedGaussExpTail.register_analytic_integral(
func=generalized_gaussexptail_integral, limits=gaussexptail_integral_limits
)