Source code for qp.utils.conversion
"""This module implements functions to convert distributions between various representations
These functions should then be registered with the `qp.ConversionDict` using `qp_add_mapping`.
That will allow the automated conversion mechanisms to work.
"""
from __future__ import annotations
import numpy as np
from scipy import integrate as sciint
from scipy import interpolate as sciinterp
from ..parameterizations.sparse_interp.sparse_rep import (
build_sparse_representation,
indices2shapes,
)
[docs]
def extract_xy_vals(in_dist: "Ensemble", xvals: np.ndarray) -> dict[str, np.ndarray]:
"""Convert using a set of x and y values.
Parameters
----------
in_dist : Ensemble
Input distributions
xvals : np.ndarray
Locations at which the pdf is evaluated
Returns
-------
data : dict[str, np.ndarray]
The extracted data
"""
yvals = in_dist.pdf(xvals)
expand_x = np.ones(yvals.shape) * np.squeeze(xvals)
return dict(xvals=expand_x, yvals=yvals)
#
# Unused conversion functions -- not tested, use at own risk
#
[docs]
def extract_fit(in_dist, **kwargs): # pragma: no cover
"""Convert to a functional distribution by fitting it to a set of x and y values
Parameters
----------
in_dist : Ensemble
Input distributions
Other Parameters
----------------
xvals : `np.array`
Locations at which the pdf is evaluated
Returns
-------
data : dict
The extracted data
"""
raise NotImplementedError("extract_fit")
# xvals = kwargs.pop('xvals', None)
# if xvals is None:
# raise ValueError("To convert using extract_fit you must specify xvals")
##vals = in_dist.pdf(xvals)
[docs]
def extract_voigt_mixmod(in_dist, **kwargs): # pragma: no cover
"""Convert to a voigt mixture model starting with a gaussian mixture model,
trivially by setting gammas to 0
Parameters
----------
in_dist : Ensemble
Input distributions
Returns
-------
data : dict
The extracted data
"""
objdata = in_dist.objdata
means = objdata["means"]
stds = objdata["stds"]
weights = objdata["weights"]
gammas = np.zeros_like(means)
return dict(means=means, stds=stds, weights=weights, gammas=gammas, **kwargs)
[docs]
def extract_voigt_xy(in_dist, **kwargs): # pragma: no cover
"""Build a voigt function basis and run a match-pursuit algorithm to fit gridded data
Parameters
----------
in_dist : Ensemble
Input distributions
Returns
-------
data : dict
The extracted data as sparse indices, basis, and metadata to rebuild the basis
"""
sparse_results = extract_voigt_xy_sparse(in_dist, **kwargs)
indices = sparse_results["indices"]
meta = sparse_results["metadata"]
w, m, s, g = indices2shapes(indices, meta)
return dict(means=m, stds=s, weights=w, gammas=g)
[docs]
def extract_voigt_xy_sparse(in_dist, **kwargs): # pragma: no cover
"""Build a voigt function basis and run a match-pursuit algorithm to fit gridded data
Parameters
----------
in_dist : Ensemble
Input distributions
Returns
-------
data : dict
The extracted data as shaped parameters means, stds, weights, gammas
"""
yvals = in_dist.objdata["yvals"]
default = in_dist.metadata["xvals"][0]
z = kwargs.pop("xvals", default)
nz = kwargs.pop("nz", 300)
minz = np.min(z)
_, j = np.where(yvals > 0)
maxz = np.max(z[j])
newz = np.linspace(minz, maxz, nz)
interp = sciinterp.interp1d(z, yvals, assume_sorted=True)
newpdf = interp(newz)
newpdf = newpdf / sciint.trapezoid(newpdf, newz).reshape(-1, 1)
ALL, bigD, _ = build_sparse_representation(newz, newpdf)
return dict(indices=ALL, metadata=bigD)