Statistical inference steps for high-energy physics analyses with JAX.
Project description
everwillow is a statistical inference library for high-energy physics built on JAX pytrees and optimistix optimizers. It provides tools for fitting, profiling, and hypothesis testing with flexible parameter handling and parameter bounds via transformations. It works with any JAX-based statistical model.
Installation
pip install everwillow
or with uv:
uv add everwillow
From source:
git clone https://github.com/MoAly98/everwillow.git
cd everwillow
uv sync
Example
A Poisson counting experiment: define a model, fit it, compute CLs p-values, and find a 95% CL upper limit — both with asymptotic formulas and toys.
import jax
import jax.numpy as jnp
import everwillow as ew
import everwillow.statelib as sl
from everwillow.hypotest.calculators import AsymptoticCalculator, HypoTestCalculator
from everwillow.hypotest.distributions import (
QTildeAsymptotic,
SimpleEmpiricalDistribution,
)
from everwillow.hypotest.test_statistics import QTilde
from everwillow.hypotest.toys import ToyGenerator
from everwillow.hypotest.upper_limit import upper_limit, upper_limit_toys
jax.config.update("jax_enable_x64", True)
# --- Model ---
signal, background = 10.0, 5.0
def nll(params, observation):
mu = params["mu"]
expected = mu * signal + background
return expected - observation["n"] * jnp.log(expected)
def predict(params_state):
mu = params_state.to_pytree()["mu"]
return {"n": mu * signal + background}
params = sl.State.from_pytree({"mu": 1.0})
observed = {"n": 12.0}
# --- Fit ---
result = ew.fit(nll_fn=nll, params=params, observation=observed)
print(result.params.to_pytree())
# {'mu': Array(0.7, dtype=float64)}
# --- Asymptotic hypothesis test ---
calc = AsymptoticCalculator(
nll_fn=nll,
params=params,
observation=observed,
poi_key="mu",
predict_fn=predict,
test_statistic=QTilde(),
distribution=QTildeAsymptotic(),
)
result = calc.test(1.0)
print(f"CLs: {calc.cls(result):.4f}")
# CLs: 0.2140
def cls_objective(poi):
return calc.cls(calc.test(poi))
limit = upper_limit(cls_objective, bounds=(0.0, 5.0), level=0.05)
print(f"95% CL upper limit (asymptotic): {float(limit):.4f}")
# 95% CL upper limit (asymptotic): 1.3673
# --- Toy-based upper limit ---
toy_gen = ToyGenerator(test_statistic=QTilde(), ntoys=5000)
def cls_from_toys(poi, key):
"""Generate toys and compute CLs at a given POI value."""
toys = toy_gen.generate(
nll,
params,
observed,
"mu",
poi_null=poi,
poi_alt=0.0,
key=key,
predict_fn=predict,
)
dist = SimpleEmpiricalDistribution.from_toys(toys)
toy_calc = HypoTestCalculator(
nll_fn=nll,
params=params,
observation=observed,
poi_key="mu",
test_statistic=QTilde(),
distribution=dist,
)
return toy_calc.cls(toy_calc.test(poi))
toy_limit = upper_limit_toys(
cls_from_toys, bounds=(0.0, 5.0), key=jax.random.key(0), tol=0.01
)
print(f"95% CL upper limit (toys): {float(toy_limit):.4f}")
Documentation
Contributing
We welcome contributions! See CONTRIBUTING.md for setup and workflow.
License
everwillow is distributed under the BSD-3-Clause License.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file everwillow-0.0.1.tar.gz.
File metadata
- Download URL: everwillow-0.0.1.tar.gz
- Upload date:
- Size: 348.3 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c70cda7585c8f3d7246183a6abdd6ccd29e184f08b568a2b1a4f2a06cf06323c
|
|
| MD5 |
004db4ea668b06a97e1c993bb852500a
|
|
| BLAKE2b-256 |
ad6f00fc4ae6b5eea67429081bdb8d2cf0b81d31efa4a671698ee05196aadaf5
|
Provenance
The following attestation bundles were made for everwillow-0.0.1.tar.gz:
Publisher:
cd.yml on MoAly98/everwillow
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
everwillow-0.0.1.tar.gz -
Subject digest:
c70cda7585c8f3d7246183a6abdd6ccd29e184f08b568a2b1a4f2a06cf06323c - Sigstore transparency entry: 1185439126
- Sigstore integration time:
-
Permalink:
MoAly98/everwillow@517efcb9628953a6b73772a8748baa4733381757 -
Branch / Tag:
refs/tags/0.1.0 - Owner: https://github.com/MoAly98
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
cd.yml@517efcb9628953a6b73772a8748baa4733381757 -
Trigger Event:
release
-
Statement type:
File details
Details for the file everwillow-0.0.1-py3-none-any.whl.
File metadata
- Download URL: everwillow-0.0.1-py3-none-any.whl
- Upload date:
- Size: 53.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
08e3fb705831f02a12f987c941084a0653d6d23381774ce1f1778c48cfd7513d
|
|
| MD5 |
494ffa904209bc1d3be74c92c817a851
|
|
| BLAKE2b-256 |
64f5103d88510a578c3ff4f046006c8c38e20be30cc352d203b6b30584703096
|
Provenance
The following attestation bundles were made for everwillow-0.0.1-py3-none-any.whl:
Publisher:
cd.yml on MoAly98/everwillow
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
everwillow-0.0.1-py3-none-any.whl -
Subject digest:
08e3fb705831f02a12f987c941084a0653d6d23381774ce1f1778c48cfd7513d - Sigstore transparency entry: 1185439140
- Sigstore integration time:
-
Permalink:
MoAly98/everwillow@517efcb9628953a6b73772a8748baa4733381757 -
Branch / Tag:
refs/tags/0.1.0 - Owner: https://github.com/MoAly98
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
cd.yml@517efcb9628953a6b73772a8748baa4733381757 -
Trigger Event:
release
-
Statement type: