Source code for zfit.models.physics

#  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 )