import sys
import jax.numpy as jnp
import numpy as np
import pandas as pd
import pymc as pm
import pymc.distributions.transforms as tr
import pytensor.tensor as at
from pytensor import shared
from pytensor.link.jax.dispatch import jax_funcify
# wrapper for pytensor tensor sorting
from pytensor.tensor.sort import SortOp
from stratmc.data import clean_data
@jax_funcify.register(SortOp)
def jax_funcify_SortOp(op, node, **kwargs):
def sort(a, axis=-1, stable=True, order=None):
return jnp.sort(a, axis = -1, stable = True)
return sort
DIST_DICT = {
"Wald": pm.Wald,
"Normal": pm.Normal,
"HalfFlat": pm.HalfFlat,
"HalfCauchy": pm.HalfCauchy,
"Uniform": pm.Uniform,
"Flat": pm.Flat,
"Beta": pm.Beta,
"Exponential": pm.Exponential,
"StudentT": pm.StudentT,
"Cauchy": pm.Cauchy,
"Laplace": pm.Laplace,
"Kumaraswamy": pm.Kumaraswamy,
"Weibull": pm.Weibull,
"HalfStudentT": pm.HalfStudentT,
"LogNormal": pm.LogNormal,
"ChiSquared": pm.ChiSquared,
"HalfNormal": pm.HalfNormal,
"Pareto": pm.Pareto,
"InverseGamma": pm.InverseGamma,
"ExGaussian": pm.ExGaussian,
"VonMises": pm.VonMises,
"SkewNormal": pm.SkewNormal,
"Triangular": pm.Triangular,
"Gumbel": pm.Gumbel,
"Logistic": pm.Logistic,
"LogitNormal": pm.LogitNormal,
"Rice": pm.Rice,
"Moyal": pm.Moyal,
"AsymmetricLaplace": pm.AsymmetricLaplace,
"PolyaGamma": pm.PolyaGamma,
}
[docs]
def build_model(sample_df, ages_df, proxies = ['d13c'], proxy_sigma_default = 0.1, approximate = False, hsgp_m = 15, hsgp_c = 1.3, ls_dist = 'Wald', ls_min = 0, ls_mu = 20, ls_lambda = 50, ls_sigma = 50, var_sigma = 10, white_noise_sigma = 1e-1, gp_mean_mu = None, gp_mean_sigma = None, offset_type = 'section', offset_prior = 'Laplace', offset_mu = 0, offset_b = 2, noise_type = 'section', noise_prior = 'HalfCauchy', noise_beta = 1, noise_sigma = None, noise_sigma_single_sample = 1, noise_sigma_studentT = 1, noise_nu = 1, jitter = 0.001, proxy_observed = True, **kwargs):
"""
Create a proxy signal (i.e., carbon isotope) :ref:`inference model <model_target>`.
Note that while excluded samples (``Exclude?`` is ``True`` in ``sample_df``) do not affect the proxy signal reconstruction, their ages are passively tracked within the inference model (if other samples from the same sections are included).
Parameters
----------
sample_df: pandas.DataFrame
:class:`pandas.DataFrame` containing proxy data for all sections. Load from .csv file using :py:meth:`load_data() <stratmc.data.load_data>` in :py:mod:`stratmc.data`.
ages_df: pandas.DataFrame
:class:`pandas.DataFrame` containing age constraints for all sections. Load from .csv file using :py:meth:`load_data() <stratmc.data.load_data>` in :py:mod:`stratmc.data`.
sections:: list(str) or numpy.array(str), optional
List of sections to include in the inference model. Defaults to all sections in ``sample_df``.
proxies: str or list(str), optional
Column or columns containing proxy data in ``sample_df``. Defaults to 'd13c'.
proxy_sigma_default: float or dict{float}, optional
Measurement uncertainty (:math:`1\\sigma`) to use for proxy observations if not specified in ``proxy_std`` column of ``sample_df``. To set a different value for each proxy, pass a dictionary with proxy names as keys. Defaults to 0.1.
approximate: bool, optional
Build model with an unapproximated latent GP (:class:`pymc.gp.Latent`) if ``False``, or a Hilbert space Gaussian process approximation (:class:`pymc.gp.HSGP`) if ``True``; defaults to ``False``. If using the HSGP approximation, also pass the ``hsgp_m`` and ``hsgp_c`` parameters (the defaults are unlikely to work well for all problems). Appropriate values for ``m`` and ``c`` can be estimated using :py:meth:`approx_hsgp_hyperparams() <pymc.gp.hsgp_approx.approx_hsgp_hyperparams>`.
hsgp_m: int or dict{int}, optional
Number of basis vectors to use in the HSGP approximation (see :class:`pymc.gp.HSGP`). Pass as a dictionary with proxy names as keys to specify a different value for each proxy. Defaults to 15.
hsgp_c: float or dict{float}, optional
Proportion extension factor for the HSGP approximation (see :class:`pymc.gp.HSGP`). Pass as a dictionary with proxy names as keys to specify a different value for each proxy. Defaults to 1.3.
ls_dist: str or dict{str}, optional
Prior distribution for the lengthscale hyperparameter of the exponential quadratic covariance kernel (:class:`pymc.gp.cov.ExpQuad <pymc.gp.cov.ExpQuad>`); set to ``Wald`` (:class:`pymc.Wald`) or ``HalfNormal`` (:class:`pymc.HalfNormal`). Defaults to ``Wald`` with ``mu = 20`` and ``lambda = 50``. To change ``mu`` and ``lambda``, pass the ``ls_mu`` and ``ls_lambda`` parameters. For ``HalfNormal``, the variance defaults to ``sigma = 50``; change by passing ``ls_sigma``. Pass as a dictionary with proxy names as keys to specify a different prior distribution for each proxy.
ls_min: float or dict{float}, optional
Minimum value for the lengthscale hyperparameter of the :class:`pymc.gp.cov.ExpQuad` covariance kernel; shifts the lengthscale prior by ``ls_min``. Pass as a dictionary with proxy names as keys to specify a different value for each proxy. Defaults to 0.
ls_mu: float or dict{float}, optional
Mean (`mu`) of the :class:`pymc.gp.cov.ExpQuad` lengthscale prior if ``ls_dist = `Wald```. Pass as a dictionary with proxy names as keys to specify a different value for each proxy. Defaults to 20.
ls_lambda: float or dict{float}, optional
Relative precision (`lam`) of the :class:`pymc.gp.cov.ExpQuad` lengthscale hyperparameter prior if ``ls_dist = `Wald```. Pass as a dictionary with proxy names as keys to specify a different value for each proxy. Defaults to 50.
ls_sigma: float or dict{float}, optional
Scale parameter (`sigma`) of the :class:`pymc.gp.cov.ExpQuad` lengthscale hyperparameter prior if ``ls_dist = `HalfNormal```. Pass as a dictionary with proxy names as keys to specify a different value for each proxy. Defaults to 50.
var_sigma: float or dict{float}, optional
Scale parameter ('sigma') of the covariance kernel variance hyperparameter prior, which is a :class:`pymc.HalfNormal` distribution. Pass as a dictionary with proxy names as keys to specify a different value for each proxy. Defaults to 10.
white_noise_sigma: float or dict{float}, optional
Amplitude of white noise component of GP covariance function. Defaults to 0.1. Should be equal to or less than the proxy measurement uncertainty; use a smaller value for proxies with low-amplitude signals (e.g., Sr isotopes). Pass as a dictionary with proxy names as keys to specify a different value for each proxy.
gp_mean_mu: float or dict{float}, optional
Mean (`mu`) of the GP mean function prior, which is a :class:`pymc.Normal` distribution. Defaults to the mean of the observations for each proxy. Pass as a dictionary with proxy names as keys to specify a different value for each proxy.
gp_mean_sigma: float or dict{float}, optional
Standard deviation (`sigma`) of the GP mean function prior, which is a :class:`pymc.Normal` distribution. Defaults to twice the standard deviation of the observations for each proxy. Pass as a dictionary with proxy names as keys to specify a different value for each proxy.
offset_type: str or dict{str}, optional
Parameterize offset such that all samples from the same section have the same offset (set to ``section``), or such that custom sample groups share an offset term (set to ``groups``, and specify the group for each sample and proxy with a ``offset_group_proxy`` column in ``sample_df``). To omit the offset term, set to ``none``. Defaults to ``section``. Pass as a dictionary with proxy names as keys to specify a different offset type for each proxy.
offset_prior: str or dict{str}, optional
Type of distribution to use for the offset prior. Pass as a ``string`` to use the same prior for all proxies, or as a ``dict`` of ``string`` (with proxy names as keys) to specify a different prior distribution for each proxy. Defaults to ``Laplace`` (:class:`pymc.Laplace`) with ``mu = 0`` and ``b = 2``. ``mu`` and ``b`` can be changed by passing ``offset_mu`` and ``offset_b``. To use other types of priors, pass the name of a distribution from :class:`pymc.distributions`, along with parameter values in ``offset_params``. Pass as a dictionary with proxy names as keys to specify a different prior for each proxy.
offset_params: dict{float} or dict{dict{float}}, optional
Only required if using a custom ``offset_prior``. Pass as a dictionary to use the same parameters for all proxies, or as a dictionary of dictionaries (with proxy names as keys) to specify different parameters for each proxy. Keys are ``param_1_name``, ``param_1``, ``param_2_name``, and ``param_2`` (with parameter names corresponding to those required for the specified :class:`pymc.distributions` object). If only one parameter is required, use ``np.nan`` for ``param_2_name`` and ``param_2``.
noise_type: str or dict{str}, optional
Parameterize noise as per-section (set to ``section``) or per-group (set to ``groups``, and specify the group for each sample and proxy with a ``noise_group_proxy`` column in ``sample_df``). Defaults to ``section``. Pass as a dictionary with proxy names as keys to specify a different noise type for each proxy.
noise_prior: str or dict{str}, optional
Type of distribution to use for the noise prior. Pass as a ``string`` to use the same prior for all proxies, or as a ``dict`` of ``string`` (with proxy names as keys) to specify a different prior distribution for each proxy. Defaults to ``HalfCauchy`` (:class:`pymc.HalfCauchy`) with ``beta = 1``. ``beta`` can be changed by passing ``noise_beta``.
Other implemented priors (note that noise must be positive-only) are``HalfStudentT`` (:class:`pymc.HalfStudentT`; defaults to ``noise_nu = 1`` and ``noise_sigma_studentT`` = 1) and ``HalfNormal`` (:class:`pymc.HalfNormal`). By default, the ``HalfNormal`` prior has ``sigma`` equal to the standard deviation of the data associated with each noise term. ``sigma`` can be changed by passing ``noise_sigma``. For sections with only 1 sample, the standard deviation of the noise prior is set by ``noise_sigma_single_sample`` (defaults to 1).
superposition_dict: dict{list(str)}, optional
Optional; dictionary specifying superposition relationships between different sections. Should only be used when superposition is not implicitly enforced by the age constraints; for example, when sections share the same minimum and maximum age constraints, but are from geological formations with a known stratigraphic relationship. Dictionary keys are section names, and the value for each key is a list of sections that must be older (stratigraphically lower).
jitter: float, optional
Value of ``jitter`` passed to :meth:`pymc.gp.Latent.prior`. Defaults to 0.001.
proxy_observed: bool, optional
Whether to pass observed values to the likelihood function; defaults to ``True``. Only set to ``False`` to generate synthetic observations from the model prior in :py:meth:`synthetic_observations_from_prior() <stratmc.synthetics>` in :py:mod:`stratmc.synthetics`.
Returns
-------
model: PyMC model
:class:`pymc.model.core.Model` object.
gp: pymc.gp.Latent or pymc.gp.HSGP
Gaussian process prior for the model. :class:`pymc.gp.Latent` if ``approximate = True``, or :class:`pymc.gp.HSGP` if ``approximate = False``.
"""
if type(proxies) == str:
proxies = list([proxies])
if type(proxy_sigma_default) != dict: #(type(proxy_sigma_default) == float) or (type(proxy_sigma_default) == int):
temp = proxy_sigma_default
proxy_sigma_default = {}
for proxy in proxies:
proxy_sigma_default[proxy] = temp
# if offset_prior passed as a string, reformat as a dictionary (keys = proxies)
if type(offset_prior) != dict:
temp = offset_prior
offset_prior = {}
for proxy in proxies:
offset_prior[proxy] = temp
# if offset_type passed as a string, reformat as a dictionary (keys = proxies)
if type(offset_type) != dict:
temp = offset_type
offset_type = {}
for proxy in proxies:
offset_type[proxy] = temp
for proxy in proxies:
if offset_type[proxy] not in ['section', 'groups', 'none']:
sys.exit(f"offset type {offset_type[proxy]} not implemented. Choose from 'section' or 'groups'.")
# also convert parameters for prior distributions to dictionaries
if type(offset_mu) != dict:
temp = offset_mu
offset_mu = {}
for proxy in proxies:
offset_mu[proxy] = temp
if type(offset_b) != dict:
temp = offset_b
offset_b = {}
for proxy in proxies:
offset_b[proxy] = temp
# if noise_prior passed as a string, reformat as a dictionary (keys = proxies)
if type(noise_prior) != dict:
temp = noise_prior
noise_prior = {}
for proxy in proxies:
noise_prior[proxy] = temp
# if noise_type passed as a string, reformat as a dictionary (keys = proxies)
if type(noise_type) != dict:
temp = noise_type
noise_type = {}
for proxy in proxies:
noise_type[proxy] = temp
for proxy in proxies:
if noise_type[proxy] not in ['section', 'groups', 'none']:
sys.exit(f"noise type {noise_type[proxy]} not implemented. Choose from 'section', 'groups', or 'none'.")
# also convert noise prior parameters to dictionaries
if type(noise_beta) != dict:
temp = noise_beta
noise_beta = {}
for proxy in proxies:
noise_beta[proxy] = temp
if type(noise_nu) != dict:
temp = noise_nu
noise_nu = {}
for proxy in proxies:
noise_nu[proxy] = temp
if type(noise_sigma_studentT) != dict:
temp = noise_sigma_studentT
noise_sigma_studentT = {}
for proxy in proxies:
noise_sigma_studentT[proxy] = temp
# create dictionaries to store offset and noise terms inside of model
offset_types = list(offset_type.values())
if ('section' in offset_types) or ('groups' in offset_types):
offset_all = {}
if 'groups' in offset_types:
offset_group_dict = {}
noise_types = list(noise_type.values())
if ('section' in noise_types) or ('groups' in noise_types):
noise_all = {}
if 'groups' in noise_types:
noise_group_dict = {}
# if only one proxy, reformat kernel parameters as dictionaries
if type(ls_dist) != dict:
temp = ls_dist
ls_dist = {}
for proxy in proxies:
ls_dist[proxy] = temp
if type(ls_mu) != dict:
temp = ls_mu
ls_mu = {}
for proxy in proxies:
ls_mu[proxy] = temp
if type(ls_lambda) != dict:
temp = ls_lambda
ls_lambda = {}
for proxy in proxies:
ls_lambda[proxy] = temp
if type(ls_sigma) != dict:
temp = ls_sigma
ls_sigma = {}
for proxy in proxies:
ls_sigma[proxy] = temp
if type(ls_min) != dict:
temp = ls_min
ls_min = {}
for proxy in proxies:
ls_min[proxy] = temp
if type(var_sigma) != dict:
temp = var_sigma
var_sigma = {}
for proxy in proxies:
var_sigma[proxy] = temp
if type(white_noise_sigma) != dict:
temp = white_noise_sigma
white_noise_sigma = {}
for proxy in proxies:
white_noise_sigma[proxy] = temp
if gp_mean_mu is not None:
if type(gp_mean_mu) != dict:
temp = gp_mean_mu
gp_mean_mu = {}
for proxy in proxies:
gp_mean_mu[proxy] = temp
else:
gp_mean_mu = {}
for proxy in proxies:
gp_mean_mu[proxy] = np.nanmean(sample_df[proxy].values)
if gp_mean_sigma is not None:
if type(gp_mean_sigma) != dict:
temp = gp_mean_sigma
gp_mean_sigma = {}
for proxy in proxies:
gp_mean_sigma[proxy] = temp
else:
gp_mean_sigma = {}
for proxy in proxies:
gp_mean_sigma[proxy] = 2 * np.nanstd(sample_df[proxy].values)
# if using HSGP approximation, reformat m and c parameters
if approximate:
if type(hsgp_m) != dict:
temp = hsgp_m
hsgp_m = {}
for proxy in proxies:
hsgp_m[proxy] = temp
if type(hsgp_c) != dict:
temp = hsgp_c
hsgp_c = {}
for proxy in proxies:
hsgp_c[proxy] = temp
for proxy in proxies:
if proxy + '_std' not in list(sample_df.columns):
sample_df[proxy + '_std'] = np.nan
idx = np.isnan(sample_df[proxy + '_std'])
sample_df.loc[idx, proxy + '_std'] = proxy_sigma_default[proxy]
if proxy + '_population_std' not in list(sample_df.columns):
sample_df[proxy + '_population_std'] = np.nan
if 'sections' in kwargs:
sections = list(kwargs['sections'])
else:
sections = list(np.unique(sample_df['section']))
if 'superposition_dict' in kwargs:
superposition_dict = kwargs['superposition_dict']
else:
superposition_dict = {}
for section in list(superposition_dict.keys()):
for older_section in superposition_dict[section]:
sections.remove(older_section)
sections.insert(0, older_section)
# clean the input DataFrames (necessary before setting noise group priors)
## instead of removing samples from dataframe if they don't have any proxy observations, mark as excluded (so the model will still keep track of age at that height)
# note - samples whose age shouldn't be tracked should simply be removed from the dataframe prior to running the inversion
# for sample_df, 'exclude' now means exclude from the likelihood calculation, but keep track of age at that height
sample_df, ages_df = clean_data(sample_df, ages_df, proxies, sections)
# ignore sections that have no observations included in the inference (samples w/ no observations were marked `Exclude? = True` in clean_data)
data_sections = list(np.unique(sample_df[~sample_df['Exclude?']]['section']))
for section in list(sections):
if section not in data_sections:
sections.remove(section)
# clean again (avoids issues w/ offset and noise groups)
sample_df, ages_df = clean_data(sample_df, ages_df, proxies, sections)
# check that for all constraints with 'shared == True', the constraint actually is used >1 time (if not, set shared = False)
for shared_age_name in ages_df[ages_df['shared?']==True]['name']:
if ages_df[(ages_df['shared?'] == True) & (ages_df['name']==shared_age_name)].shape[0] < 2:
idx = ages_df[(ages_df['shared?'] == True) & (ages_df['name']==shared_age_name)].index
ages_df.loc[idx, 'shared?'] = False
# by default, set noise prior to HalfNormal with sigma equal to the standard deviation of observations from the group or section. if a noise_sigma value is passed by the user, use the specified noise prior instead
if noise_sigma is not None:
if type(noise_sigma) != dict:
temp = noise_sigma
noise_sigma = {}
for proxy in proxies:
if noise_type[proxy] == 'groups':
noise_groups = np.array(sample_df[(~np.isnan(sample_df[proxy])) & (~sample_df['Exclude?'])]['noise_group_' + proxy].unique()).astype(str)
noise_sigma[proxy] = {}
for group in noise_groups:
noise_sigma[proxy][group] = temp
if noise_type[proxy] == 'section':
noise_sigma[proxy] = {}
for section in sections:
noise_sigma[proxy][section] = temp
# calculate observation standard deviations
else:
noise_sigma = {}
for proxy in proxies:
if noise_type[proxy] == 'groups':
noise_groups = np.array(sample_df[(~np.isnan(sample_df[proxy])) & (~sample_df['Exclude?'])]['noise_group_' + proxy].unique()).astype(str)
noise_sigma[proxy] = {}
for group in noise_groups:
if sample_df[(sample_df['noise_group_' + proxy] == group) & ~(sample_df['Exclude?']) & ~(np.isnan(sample_df[proxy]))].shape[0] > 1:
noise_sigma[proxy][group] = np.nanstd(sample_df[(sample_df['noise_group_' + proxy] == group) & ~(sample_df['Exclude?'])][proxy].values)
# if only 1 sample, set noise_sigma equal to a prescribed constant
else:
noise_sigma[proxy][group] = noise_sigma_single_sample # np.abs(sample_df[(sample_df['noise_group_' + proxy] == group) & ~(sample_df['Exclude?'])][proxy].values[0]/2)
elif noise_type[proxy] == 'section':
noise_sigma[proxy] = {}
for section in sections:
if sample_df[(sample_df['section'] == section) & ~(sample_df['Exclude?']) & ~(np.isnan(sample_df[proxy]))].shape[0] > 1:
noise_sigma[proxy][section] = np.nanstd(sample_df[(sample_df['section'] == section) & ~(sample_df['Exclude?'])][proxy].values)
# if only 1 sample, set noise_sigma equal to a prescribed constant
else:
noise_sigma[proxy][section] = noise_sigma_single_sample # np.abs(sample_df[(sample_df['section'] == section) & ~(sample_df['Exclude?'])][proxy].values[0]/2)
if 'offset_params' in kwargs:
offset_params = kwargs['offset_params']
# if not a dictionary, throw an error
if type(offset_params) != dict:
sys.exit(f"offset_params must be a dictionary")
# if it's a dictionary, check that it has all the required items
else:
offset_keys = list(offset_params.keys())
# if passed correctly, just reformat to assign same dict to each proxy
if ('param_1_name' in offset_keys) & ('param_1' in offset_keys) & ('param_2_name' in offset_keys) & ('param_2' in offset_keys):
temp_dict = {}
for proxy in proxies:
temp_dict[proxy] = offset_params.copy()
offset_params = temp_dict.copy()
# if not passed correctly, and also not passed with proxies as keys
elif all([proxy not in offset_keys for proxy in proxies]):
sys.exit(f"offset_params must be a dictionary (or dictionary of dictionaries, if specifying different prior parameters for each proxy) with keys 'param_1_name','param_1', 'param_2_name', and 'param_2'")
# if passed with proxies as keys
else:
for proxy in proxies:
# throw an error if every proxy doesn't have its own dictionary (unless using the default Laplace prior)
if proxy not in offset_keys:
# if using offset_mu and offset_b for the laplace prior, set prior_params dictionary to None
if offset_prior[proxy] == 'Laplace':
offset_params[proxy] = None
else:
sys.exit(f"offset_params must be a dictionary (or dictionary of dictionaries, if specifying different prior parameters for each proxy) with keys 'param_1_name','param_1', 'param_2_name', and 'param_2'. If passing with proxy names as keys, make sure that all parameters have been assigned to every proxy (except those using a Laplace prior with parameters offset_mu and offset_b).")
elif (proxy in offset_keys) and (offset_prior[proxy] == 'Laplace'):
print('Note: offset parameters passed in offset_params supersede the offset_mu and offset_b arguments')
# if the proxy has a dictionary, check that all the required parameters have been passed
else:
offset_proxy_keys = list(offset_params[proxy].keys())
if ('param_1_name' not in offset_proxy_keys) | ('param_1' not in offset_proxy_keys) | ('param_2_name' not in offset_proxy_keys) | ('param_2' not in offset_proxy_keys):
sys.exit(f"offset_params must be a dictionary (or dictionary of dictionaries, if specifying different prior parameters for each proxy) with keys 'param_1_name','param_1', 'param_2_name', and 'param_2'. If passing with proxy names as keys, make sure that all parameters have been assigned to every proxy.")
else:
offset_params = {}
for proxy in proxies:
offset_params[proxy] = None
with pm.Model() as model:
gp_ls = {}
gp_var = {}
m_proxy = {}
mean_fun = {}
cov1 = {}
cov2 = {}
gp = {}
# GP for each proxy
for proxy in proxies:
if ls_dist[proxy] == 'Wald':
ls_temp = pm.Wald('gp_ls_unshifted_' + proxy, mu = ls_mu[proxy], lam = ls_lambda[proxy], shape = 1)
gp_ls[proxy] = pm.Deterministic('gp_ls_' + proxy, ls_temp + ls_min[proxy])
elif ls_dist[proxy] == 'HalfNormal':
ls_temp = pm.HalfNormal('gp_ls_unshifted_' + proxy, sigma = ls_sigma[proxy], shape = 1)
gp_ls[proxy] = pm.Deterministic('gp_ls_' + proxy, ls_temp + ls_min[proxy])
gp_var[proxy] = pm.HalfNormal('gp_var_' + proxy, sigma = var_sigma[proxy], shape=1)
m_proxy[proxy] = pm.Normal('m_' + proxy, mu = gp_mean_mu[proxy], sigma = gp_mean_sigma[proxy], shape = 1) #
# mean and covariance functions
mean_fun[proxy] = pm.gp.mean.Constant(m_proxy[proxy])
cov1[proxy] = gp_var[proxy] ** 2 * pm.gp.cov.ExpQuad(1, gp_ls[proxy])
cov2[proxy] = pm.gp.cov.WhiteNoise(white_noise_sigma[proxy])
# GP prior
if not approximate:
gp[proxy] = pm.gp.Latent(mean_func = mean_fun[proxy], cov_func = cov1[proxy] + cov2[proxy])
if approximate:
print('Using HSGP approximation for ', proxy)
gp[proxy] = pm.gp.HSGP(m = [hsgp_m[proxy]], c = hsgp_c[proxy], mean_func = mean_fun[proxy], cov_func = cov1[proxy])
ages_all = []
proxy_all = {}
proxy_sigma_all = {}
proxy_population_std_all = {}
include_idx_all = {}
for proxy in proxies:
proxy_all[proxy] = []
include_idx_all[proxy] = []
proxy_sigma_all[proxy] = []
proxy_population_std_all[proxy] = []
if noise_type[proxy] != 'none':
noise_all[proxy] = []
if offset_type[proxy] != 'none':
offset_all[proxy] = []
# create distribution objects for each unique shared constraint
# in this implementation, constraints should only be labeled as 'shared?' if diachronous behavior is not allowed -- constraints that are the same (e.g.,
# a fossil first appearanace date), but that may be diachronous between sections, should be set to shared = False
shared_constraints = {}
shared_ages = ages_df[ages_df['shared?']==True]
if len(shared_ages) > 0:
unique_shared_constraints = np.unique(shared_ages['name'])
for constraint in unique_shared_constraints:
constraint = str(constraint)
constraint_df = shared_ages[shared_ages['name']==constraint]
dist = np.unique(constraint_df['distribution_type'])
dist_age = np.unique(constraint_df['age'])
dist_age_std = np.unique(constraint_df['age_std'])
if (len(dist) > 1) or (len(dist_age) > 1) or (len(dist_age_std) > 1):
sys.exit(f"Initialization of shared age constraint {constraint} is inconsistent. Check that distribution type and parameters are the same for each section.")
dist = dist[0]
if dist == 'Normal':
shared_constraints[constraint] = pm.Normal(constraint, mu = dist_age[0], sigma = dist_age_std[0])
else:
# if not implemented, throw error
if dist not in DIST_DICT.keys():
sys.exit(f"{dist} distribution not implemented. Add to DIST_DICT or choose a different distribution.")
dist_args = {}
param_1 = constraint_df['param_1'].values[0]
param_1_name = constraint_df['param_1_name'].values[0]
param_2 = constraint_df['param_2'].values[0]
param_2_name = constraint_df['param_2_name'].values[0]
if not pd.isna(param_1):
dist_args[param_1_name] = param_1
if not pd.isna(param_2):
dist_args[param_2_name] = param_2
shared_constraints[constraint] = DIST_DICT[dist](constraint, **dist_args)
# if offset and/or noise terms are manually grouped by the user, create distribution for each group
# separate offset and term for each proxy
for proxy in proxies:
if offset_type[proxy] == 'groups':
offset_group_dict[proxy] = {}
offset_groups = np.array(sample_df[~np.isnan(sample_df[proxy]) & (~sample_df['Exclude?'])]['offset_group_' + proxy].unique()).astype(str)
for group in offset_groups:
# if using the default laplace prior, and offset_b and offset_mu aren't superseded by parameters passed in offset_params dict
if (offset_prior[proxy] == 'Laplace') and (offset_params[proxy] is None):
offset_group_dict[proxy][group] = pm.Laplace(group + '_group_offset_' + proxy, mu = offset_mu[proxy], b = offset_b[proxy], shape = 1)
offset_likelihood = pm.Laplace(group + '_group_offset_likelihood_' + proxy, mu = offset_group_dict[proxy][group], b = offset_b[proxy], observed = np.array([0]))
else:
# if not implemented, throw error
if offset_prior[proxy] not in DIST_DICT.keys():
sys.exit(f"{offset_prior[proxy]} distribution not implemented. Add to DIST_DICT or choose a different distribution for `offset_prior`, {proxy}.")
dist_args = {}
param_1 = offset_params[proxy]['param_1']
param_1_name = offset_params[proxy]['param_1_name']
param_2 = offset_params[proxy]['param_2']
param_2_name = offset_params[proxy]['param_2_name']
if not pd.isna(param_1):
dist_args[param_1_name] = param_1
if not pd.isna(param_2):
dist_args[param_2_name] = param_2
print('Note that offset likelihood function (which explicitly rewards offset = 0) is not implemented for custom prior distributions.')
offset_group_dict[proxy][group] = DIST_DICT[offset_prior[proxy]](group + '_group_offset_' + proxy, **dist_args, shape = 1)
if noise_type[proxy] == 'groups':
noise_group_dict[proxy] = {}
noise_groups = np.array(sample_df[(~np.isnan(sample_df[proxy])) & (~sample_df['Exclude?'])]['noise_group_' + proxy].unique()).astype(str)
for group in noise_groups:
if noise_prior[proxy] == 'HalfCauchy':
noise_group_dict[proxy][group] = pm.HalfCauchy(str(group) + '_group_noise_' + proxy, beta = noise_beta[proxy], shape = 1)
elif noise_prior[proxy] == 'HalfNormal':
noise_group_dict[proxy][group] = pm.HalfNormal(str(group) + '_group_noise_' + proxy, sigma = noise_sigma[proxy][group], shape = 1)
elif noise_prior[proxy] == 'HalfStudentT':
noise_group_dict[proxy][group] = pm.HalfStudentT(str(group) + '_group_noise_' + proxy, nu = noise_nu[proxy], sigma = noise_sigma_studentT[proxy], shape = 1)
else:
# throw an error if not HalfCauchy, HalfNormal, or HalfStudentT -- custom noise priors not allowed (since noise prior must be one of the positive-only distributions)
sys.exit(f"{noise_prior[proxy]} noise prior not implemented. Options are HalfCauchy (default), HalfNormal, or HalfStudentT.")
# dictionary to store sample age distributions for each section -- required for superposition between sections
section_age_dist = {}
for section in sections:
intervals = []
ages = []
section_df = sample_df[sample_df['section']==section]
# separate dataframes for intermediate detrital or intrusive constraints and depositional age constraints
section_age_df = ages_df[(ages_df['section']==section) & (~ages_df['intermediate detrital?']) & (~ages_df['intermediate intrusive?']) & (~ages_df['depositional?'])]
intermediate_detrital_section_ages_df = ages_df[(ages_df['section']==section) & (ages_df['intermediate detrital?'])]
intermediate_intrusive_section_ages_df = ages_df[(ages_df['section']==section) & (ages_df['intermediate intrusive?'])]
depositional_section_ages_df = ages_df[(ages_df['section']==section) & (ages_df['depositional?'])]
section_ages = section_age_df['age'].values
section_ages_unc = section_age_df['age_std'].values
heights = section_df['height'].values
age_heights = section_age_df['height'].values
# grab list of depositional age constraint names for current section (must match name of age in ages_df)
depositional_age_names = section_df['depositional age'].dropna().unique()
# create age constraint distributions
if len(age_heights) == 0:
print(str(section) + ' has no age constraints')
else:
ages = {}
label = str(section) +'_'
# the input ages (the means) need to be in stratigraphic superposition, but ordered transform is still used so that the posteriors cannot be out of superposition due to 2sigma uncertainty -- avoids potentially bad initialization
# biuld distributions if all age constraint priors are Gaussian and not shared
if all(section_age_df['distribution_type']=='Normal') and all(section_age_df['shared?']==False):
# for initvals, using np.sort instead of np.flip to account for scenario where means of reported ages are out of superposition
radiometric_age_flip = pm.Normal(label + 'flip_radiometric_age',
mu = np.flip(section_ages), # youngest to oldest
sigma = np.flip(section_ages_unc),
shape = section_ages.shape,
transform = tr.Ordered(),
initval = np.sort(section_ages))
radiometric_age_tensor = pm.Deterministic(label + 'radiometric_age', np.flip(radiometric_age_flip)) # flipping back to oldest --> youngest
else:
print('Using radiometric age priors specified in ages_df for section ' + str(section))
radiometric_age = {}
age_dist_names = []
for i in np.arange(len(section_ages)):
label = str(section) + '_' + str(i) + '_'
dist = section_age_df['distribution_type'].values[i]
constraint_shared = section_age_df['shared?'].values[i]
# for shared constraints, link to existing distribution
if constraint_shared == True:
shared_constraint_name = section_age_df['name'].values[i]
radiometric_age[i] = shared_constraints[shared_constraint_name]
age_dist_names.append(shared_constraint_name)
# if the constraint is not shared, build a new distribution
else:
if dist == 'Normal':
radiometric_age[i] = pm.Normal(label + 'radiometric_age', mu = section_ages[i], sigma = section_ages_unc[i])
age_dist_names.append(label + 'radiometric_age')
else:
# if not implemented, throw error
if dist not in DIST_DICT.keys():
sys.exit(f"{dist} distribution not implemented. Add to DIST_DICT or choose a different distribution.")
dist_args = {}
param_1 = section_age_df['param_1'].values[i]
param_1_name = section_age_df['param_1_name'].values[i]
param_2 = section_age_df['param_2'].values[i]
param_2_name = section_age_df['param_2_name'].values[i]
if not pd.isna(param_1):
dist_args[param_1_name] = param_1
if not pd.isna(param_2):
dist_args[param_2_name] = param_2
radiometric_age[i] = DIST_DICT[dist](label + 'radiometric_age', **dist_args)
age_dist_names.append(label + 'radiometric_age')
# make a vector of all the radiometric ages for superposition function
radiometric_age_tensor = at.zeros((len(section_ages),))
for i in np.arange(len(section_ages)):
radiometric_age_tensor = at.set_subtensor(radiometric_age_tensor[i], radiometric_age[i])
label = str(section) +'_'
radiometric_age_tensor = pm.Deterministic(label + 'radiometric_age', radiometric_age_tensor)
superposition(radiometric_age_tensor, age_dist_names, model, section_age_df, section)
# if there are samples below the basal age constraint, throw an error
if (heights[0] < age_heights[0]):
sys.exit(f"Section {section} does not have a basal age constraint. Add a maximum section age to ages_df.")
# if the section has no upper age constraint, throw an error
if heights[-1] >= age_heights[-1]:
sys.exit(f"Section {section} does not have an upper age constraint. Add a minimum section age to ages_df.")
# create sample age distributions for section (by interval)
for interval in np.arange(0, len(age_heights)-1).tolist():
label = str(section)+'_'+ str(interval) +'_'
above = section_df['height']>=age_heights[interval]
below = section_df['height']<age_heights[interval+1]
interval_df = section_df[above & below]
# interval_samples = interval_df[proxy].values
interval_superposition = interval_df['superposition?'].values
interval_heights = interval_df['height'].values
above = intermediate_detrital_section_ages_df['height']>age_heights[interval]
below = intermediate_detrital_section_ages_df['height']<age_heights[interval+1]
detrital_interval_df = intermediate_detrital_section_ages_df[above & below]
above = intermediate_intrusive_section_ages_df['height']>age_heights[interval]
below = intermediate_intrusive_section_ages_df['height']<age_heights[interval+1]
intrusive_interval_df = intermediate_intrusive_section_ages_df[above & below]
# if there are samples in the current interval
# if interval = 0, and the current section is in the superposition dictionary, then use the minimum age from sections that must be older as the maximum age for the current section
# note: if appropriate, make sure that the overlying age constraint is shared between the sections to avoid potential superposition issues
if len(interval_heights) > 0:
if section in list(superposition_dict.keys()):
base_age_dist = pm.math.concatenate([section_age_dist[older_section] for older_section in superposition_dict[section]]).min()
else:
base_age_dist = radiometric_age_tensor[interval]
observed_age_diff = pm.Deterministic(label + 'obs_age_diff', base_age_dist - radiometric_age_tensor[interval+1])
# sort random draws if >1 sample
if len(interval_heights) > 1:
shuffle_heights = np.unique(interval_heights[~interval_superposition])
random_sample_ages_unsorted = pm.Uniform(label + 'unsorted_random_ages', lower = 0, upper = 1, size = len(interval_heights))
# random_sample_ages_unsorted = pm.Deterministic(label + 'unsorted_random_ages_scaled', random_sample_ages_unsorted_unscaled/1e6)
# if superposition is known for all samples, sort random ages
if all(interval_superposition):
random_sample_ages = pm.Deterministic(label + 'random_ages', at.sort(random_sample_ages_unsorted))
# if there's no superposition information for any samples, skip sorting
elif (all(~interval_superposition)) and (len(shuffle_heights) == 1):
random_sample_ages = pm.Deterministic(label + 'random_ages', random_sample_ages_unsorted)
# if only some samples are missing superposition information, only sort samples w/ superposition = True
else:
sorted_idx = at.argsort(random_sample_ages_unsorted)
# create a dictionary to store lists of indices to shuffle (one list per stratigraphic horizon)
shuffle_group_idx = {}
for h in shuffle_heights:
shuffle_group_idx[h] = []
# grab indices for each group of unsorted samples
for i, h in enumerate(interval_heights):
if h in shuffle_heights:
shuffle_group_idx[h].append(i)
# sort all of the ages
random_sample_ages = at.sort(random_sample_ages_unsorted)
# unsort each group of samples without superposition information (base to top)
for i, h in enumerate(shuffle_heights):
# replace w/ another set of random draws from Uniform(0, 1), then scale between bounding samples
# note - not possible to re-shuffle the original draws (permutations not allowed in logp graph)
interval_shuffled_random_ages = pm.Uniform('shuffled_ages_' + str(section) + '_' + str(h),
lower = 0,
upper = 1,
size = len(shuffle_group_idx[h])
)
# get indices of samples below and above the shuffled range
start_shuffle_idx = np.max([0, shuffle_group_idx[h][0]-1])
stop_shuffle_idx = np.min([len(interval_heights) - 1, shuffle_group_idx[h][-1] + 1])
shuffle_base_age = random_sample_ages[start_shuffle_idx]
shuffle_top_age = random_sample_ages[stop_shuffle_idx]
# check if the starting index is also from a previously shuffled interval. if yes, reset base age to youngest sample in that group
# don't need to worry about the overlying interval, because it hasn't been reset yet
if i != 0:
start_shuffle_height = interval_heights[start_shuffle_idx]
# if the underlying sample was also shuffled, find the youngest sample in its group, and use it as the new 'base age'
if start_shuffle_height in shuffle_heights[:i]:
under_shuffle_idx = shuffle_group_idx[start_shuffle_height]
# note: because of how the scaled ages are calculated (age = base - unscaled age), the youngest sample will have the highest value in [0, 1]
shuffle_base_age = at.max(random_sample_ages[under_shuffle_idx])
# calculate (unscaled) total time spanned by the shuffled interval
# observed_shuffle_age_diff = random_sample_ages[stop_shuffle_idx] - random_sample_ages[start_shuffle_idx]
# note: ages still flipped s.t. larger values = younger
observed_shuffle_age_diff = shuffle_top_age - shuffle_base_age
# scale the shuffled [0, 1] ages s.t. the values fall in between the (sorted) values for over/underlying samples
# scaling: max - (random * observed diff)
interval_shuffled_scaled_random_ages = pm.Deterministic('shuffled_scaled_ages_' + str(section) + '_' + str(h),
shuffle_top_age - (interval_shuffled_random_ages * observed_shuffle_age_diff))
# insert unsorted random ages in tensor
random_sample_ages = at.set_subtensor(random_sample_ages[shuffle_group_idx[h]], interval_shuffled_scaled_random_ages)
# store final random age tensor in deterministic
random_sample_ages = pm.Deterministic(label + 'random_ages', random_sample_ages)
# skip sorting if interval only contains 1 sample
else:
random_sample_ages = pm.Uniform(label + 'random_ages', lower = 0, upper = 1, size = len(interval_heights))
# scaled age parameterization
scaling_factor_1 = pm.Uniform(label + 'scaling_factor_1', lower = 0, upper = 1, size = 1)
scaling_factor_2 = pm.Uniform(label + 'scaling_factor_2', lower = 0, upper = 1, size = 1)
# scaled_ages = max - random_ages * age_range * sf1 - (1 - sf2) * age_range
ages[interval] = pm.Deterministic(label + 'ages', base_age_dist - (random_sample_ages * observed_age_diff * scaling_factor_1) - (1 - scaling_factor_2) * observed_age_diff * (1 - scaling_factor_1))
intervals.append(interval)
# if there are intermediate detrital ages in the section, check if they're inside the current interval (iterate over constraints)
detrital_age_dist_names = []
detrital_age_dist_names_radio = []
if detrital_interval_df.shape[0] > 0:
# enforce detrital ages -- note that initial values will be set base --> top
for i in np.arange(detrital_interval_df.shape[0]):
# if there are overlying samples in interval, enforce maximum age
# construct DZ age prior
dist = detrital_interval_df['distribution_type'].values[i]
constraint_shared = detrital_interval_df['shared?'].values[i]
if constraint_shared == True:
shared_constraint_name = detrital_interval_df['name'].values[i]
intermediate_detrital_age = shared_constraints[shared_constraint_name]
intermediate_detrital_age_dist_name = shared_constraint_name
else:
if dist == 'Normal':
intermediate_detrital_age = pm.Normal(label + 'detrital_age_' + str(i),
mu = detrital_interval_df['age'].values[i],
sigma = detrital_interval_df['age_std'].values[i])
intermediate_detrital_age_dist_name = label + 'detrital_age_' + str(i)
else:
# if distribution not implemented, throw error
if dist not in DIST_DICT.keys():
sys.exit(f"{dist} distribution not implemented. Add to DIST_DICT or choose a different distribution.")
dist_args = {}
param_1 = detrital_interval_df['param_1'].values[i]
param_1_name = detrital_interval_df['param_1_name'].values[i]
param_2 = detrital_interval_df['param_2'].values[i]
param_2_name = detrital_interval_df['param_2_name'].values[i]
if not pd.isna(param_1):
dist_args[param_1_name] = param_1
if not pd.isna(param_2):
dist_args[param_2_name] = param_2
intermediate_detrital_age = DIST_DICT[dist](label + 'detrital_age_' + str(i), **dist_args)
intermediate_detrital_age_dist_name = label + 'detrital_age_' + str(i)
# if there are samples above the detrital age, enforce with potential
if len(interval_df[interval_df['height']>=detrital_interval_df['height'].values[i]]['height'].values)>0:
detrital_age_dist_names.append(intermediate_detrital_age_dist_name)
detrital_age_dist_names_radio.append(intermediate_detrital_age_dist_name)
# enforce detrital age with pm.Potential
intermediate_detrital_potential(intermediate_detrital_age,
intermediate_detrital_age_dist_name,
ages[interval],
interval_df['height'].values,
detrital_interval_df['height'].values[i],
section
)
# variables:
# dz age: intermediate_detrital_age
# dz age name: intermediate_detrital_age_dist_name
# sample ages (unsorted random draws U[0, 1]): random_sample_ages_unsorted
# sample age dist name: label + 'unsorted_random_ages'
# base age dist name: see above
# upper age dist name: see above
# scaling factor 1 name: label + 'scaling_factor_1'
# scaling factor 2 name: label + 'scaling_factor_2'
# if there are no samples above the current detrital constraint, just add its name to list of detritals that apply to overlying depositional age constraint
else:
detrital_age_dist_names_radio.append(intermediate_detrital_age_dist_name)
# if there are intermediate intrusive ages in section, check if they're inside this interval
intrusive_age_dist_names = []
intrusive_age_dist_names_radio = []
if intrusive_interval_df.shape[0] > 0:
# enforce intrusive ages -- note that initial values will be set base --> top
for i in np.arange(intrusive_interval_df.shape[0]):
# construct intrusive age prior
dist = intrusive_interval_df['distribution_type'].values[i]
constraint_shared = intrusive_interval_df['shared?'].values[i]
if constraint_shared == True:
shared_constraint_name = intrusive_interval_df['name'].values[i]
intermediate_intrusive_age = shared_constraints[shared_constraint_name]
intermediate_intrusive_age_dist_name = shared_constraint_name
else:
if dist == 'Normal':
intermediate_intrusive_age = pm.Normal(label + 'intrusive_age_' + str(i),
mu = intrusive_interval_df['age'].values[i],
sigma = intrusive_interval_df['age_std'].values[i])
intermediate_intrusive_age_dist_name = label + 'intrusive_age_' + str(i)
else:
# if distribution not implemented, throw error
if dist not in DIST_DICT.keys():
sys.exit(f"{dist} distribution not implemented. Add to DIST_DICT or choose a different distribution.")
dist_args = {}
param_1 = intrusive_interval_df['param_1'].values[i]
param_1_name = intrusive_interval_df['param_1_name'].values[i]
param_2 = intrusive_interval_df['param_2'].values[i]
param_2_name = intrusive_interval_df['param_2_name'].values[i]
if not pd.isna(param_1):
dist_args[param_1_name] = param_1
if not pd.isna(param_2):
dist_args[param_2_name] = param_2
intermediate_intrusive_age = DIST_DICT[dist](label + 'intrusive_age_' + str(i), **dist_args)
intermediate_intrusive_age_dist_name = label + 'intrusive_age_' + str(i)
# if there are samples below intrusive age, enforce with potential
if len(interval_df[interval_df['height']<=intrusive_interval_df['height'].values[i]]['height'].values)>0:
intrusive_age_dist_names.append(intermediate_intrusive_age_dist_name)
intrusive_age_dist_names_radio.append(intermediate_intrusive_age_dist_name)
intermediate_intrusive_potential(intermediate_intrusive_age,
intermediate_intrusive_age_dist_name,
ages[interval],
interval_df['height'].values,
intrusive_interval_df['height'].values[i],
section,
)
# if there are no samples beneath the current intrusive constraint, just add its name to list of intrusives that apply to underlying depositional age constraint
else:
intrusive_age_dist_names_radio.append(intermediate_intrusive_age_dist_name)
if (detrital_interval_df.shape[0] > 0) or (intrusive_interval_df.shape[0] > 0):
if all(section_age_df['distribution_type']=='Normal') and all(section_age_df['shared?']==False):
# THESE WILL BE UPSIDE DOWN -- inside of the intrusive potential, use this name, but flip the initial values
# with np.flip() THEN index with [interval] and [interval+1]
base_age_dist_name = str(section) +'_' + 'flip_radiometric_age'
upper_age_dist_name = str(section) +'_' + 'flip_radiometric_age'
shared_radiometric_age_dist = True
base_age_idx = interval
top_age_idx = interval + 1
else:
base_age_dist_name = age_dist_names[interval]
upper_age_dist_name = age_dist_names[interval + 1]
shared_radiometric_age_dist = False
base_age_idx = None
top_age_idx = None
if len(intrusive_age_dist_names_radio) > 0:
# enforce superposition between basal age constraint and intrusive ages
# NOTE: base_age_dist has to be flipped before using index
superposition_depositional_and_limiting_ages(model, base_age_dist_name, [], intrusive_age_dist_names_radio, section, depositional_age_idx = base_age_idx)
if len(detrital_age_dist_names_radio) > 0:
# enforce superposition between top age constriant and detrital ages
# NOTE: upper_age_dist has to be flipped before using index inside of function
superposition_depositional_and_limiting_ages(model, upper_age_dist_name, detrital_age_dist_names_radio, [], section, depositional_age_idx = top_age_idx)
if (len(detrital_age_dist_names) > 0) or (len(intrusive_age_dist_names) > 0):
if len(interval_heights) > 1:
age_label = 'unsorted_random_ages'
else:
age_label = 'random_ages'
get_valid_initial_ages(detrital_age_dist_names,
intrusive_age_dist_names,
base_age_dist_name,
upper_age_dist_name,
label + age_label,
interval_df['height'].values,
detrital_interval_df['height'].values,
intrusive_interval_df['height'].values,
model,
interval,
sf1_name = label + 'scaling_factor_1',
sf2_name = label + 'scaling_factor_2',
shared_radiometric_age_dist = shared_radiometric_age_dist)
label = str(section) + '_'
# concatenate ages from all intervals in section
ages = [ages[interval] for interval in intervals]
section_age_tensor = at.zeros((len(heights),))
count = 0
for age_sub in ages:
for i in np.arange(0, age_sub.shape.eval()[0]):
section_age_tensor = at.set_subtensor(section_age_tensor[count], age_sub[i])
count += 1
section_age_dist[section] = pm.Deterministic(label+'ages', section_age_tensor)
# for samples with depositional ages, enforce with a likelihood function
if len(depositional_age_names) > 0:
for constraint in depositional_age_names:
print(f'Adding depositional age likelihood term for section {section}: {constraint}')
age_mu = np.unique(depositional_section_ages_df[depositional_section_ages_df['name'] == constraint]['age'])
age_std = np.unique(depositional_section_ages_df[depositional_section_ages_df['name'] == constraint]['age_std'])
if (len(age_mu) > 1) or (len(age_std) > 1):
sys.exit(f"Initialization of depositional age constraint {constraint} is inconsistent. Check that the mean and standard deviation are the same for each instance in the ages DataFrame.")
elif (len(age_mu) == 0) or (len(age_std) == 0):
sys.exit(f"Depositional age constraint {constraint} not included in age constraint DataFrame for section {section}. Check that the constraint name in ages_df matches the name in sample_df, and that the constraint has not been excluded.")
else:
# grab indices of samples with the current depositional age
dep_constraint_idx = np.where(section_df['depositional age'] == constraint)[0]
# likelihood function: mean = modeled sample age, sigma = depositioanl age constraint standard deviation, observed = depositional age constraint mean
dep_age_dist = pm.Normal(str(section) + '_depositional_age_likelihood_' + constraint,
mu = section_age_dist[section][dep_constraint_idx],
sigma = list([age_std[0]]) * len(dep_constraint_idx),
observed = list([age_mu[0]]) * len(dep_constraint_idx)
)
ages_all = np.append(ages_all, ages)
proxy_obs = {}
include_idx = {}
proxy_sigma_obs = {}
proxy_population_std_obs = {}
if ('section' in noise_types) or ('groups' in noise_types):
section_noise = {}
if ('section' in offset_types) or ('groups' in offset_types):
section_offset = {}
# set up the noise and offset terms, and update the proxy observation lists
for proxy in proxies:
# ignore samples marked 'exclude' + nans (used for indexing later -- for now, concatenate all values)
include_idx[proxy] = (~section_df['Exclude?'].values.astype(bool)) & (~np.isnan(section_df[proxy]))
proxy_obs[proxy] = section_df[proxy].values
proxy_sigma_obs[proxy] = section_df[proxy + '_std'].values
proxy_population_std_obs[proxy] = section_df[proxy + '_population_std'].values
if offset_type[proxy] == 'section':
# if using the default Laplace prior, and offset_b and offset_mu aren't superseded by parameters passed in offset_params dict
if (offset_prior[proxy] == 'Laplace') and (offset_params[proxy] is None):
section_offset[proxy] = pm.Laplace(label + 'section_offset_' + proxy, mu = offset_mu[proxy], b = offset_b[proxy], shape = 1)
offset_likelihood = pm.Laplace(label + "section_offset_likelihood_" + proxy, mu = section_offset[proxy], b = offset_b[proxy], shape = 1, observed = np.array([0]))
else:
# if requested offset distribution not implemented, throw error
if offset_prior[proxy] not in DIST_DICT.keys():
sys.exit(f"{offset_prior[proxy]} distribution not implemented. Add to DIST_DICT or choose a different distribution for `offset_prior`, {proxy}.")
dist_args = {}
param_1 = offset_params[proxy]['param_1']
param_1_name = offset_params[proxy]['param_1_name']
param_2 = offset_params[proxy]['param_2']
param_2_name = offset_params[proxy]['param_2_name']
if not pd.isna(param_1):
dist_args[param_1_name] = param_1
if not pd.isna(param_2):
dist_args[param_2_name] = param_2
print('Note that offset likelihood function (which explicitly rewards offset = 0) is not implemented for custom prior distributions.')
section_offset[proxy] = DIST_DICT[offset_prior[proxy]](label + 'section_offset_' + proxy, **dist_args, shape = 1)
# only create offset distributions for samples that will be included in the likelihood function
offset_all[proxy].append(([1] * len(heights[include_idx[proxy]])) * section_offset[proxy])
elif offset_type[proxy] == 'groups':
section_offset[proxy] = []
offset_group_list = section_df['offset_group_' + proxy][include_idx[proxy]].values
for offset_key in offset_group_list:
section_offset[proxy].append(offset_group_dict[proxy][offset_key])
offset_all[proxy].append(section_offset[proxy])
if noise_type[proxy] == 'section':
if noise_prior[proxy] =='HalfCauchy':
section_noise[proxy] = pm.HalfCauchy(label + 'section_noise_' + proxy, beta = noise_beta[proxy], shape = 1) # beta = 2
elif noise_prior[proxy] == 'HalfNormal':
section_noise[proxy] = pm.HalfNormal(label + 'section_noise_' + proxy, sigma = noise_sigma[proxy][section], shape = 1)
elif noise_prior[proxy] == 'HalfStudentT':
section_noise[proxy] = pm.HalfStudentT(label + 'section_noise_' + proxy, nu = noise_nu[proxy], sigma = noise_sigma_studentT[proxy], shape = 1)
noise_all[proxy].append(([1] * len(heights[include_idx[proxy]])) * section_noise[proxy])
elif noise_type[proxy] == 'groups':
section_noise[proxy] = []
noise_group_list = section_df['noise_group_' + proxy][include_idx[proxy]].values
for noise_key in noise_group_list:
section_noise[proxy].append(noise_group_dict[proxy][noise_key])
noise_all[proxy].append(section_noise[proxy])
proxy_all[proxy] = np.append(proxy_all[proxy], proxy_obs[proxy])
include_idx_all[proxy] = np.append(include_idx_all[proxy], include_idx[proxy])
proxy_sigma_all[proxy] = np.append(proxy_sigma_all[proxy], proxy_sigma_obs[proxy])
proxy_population_std_all[proxy] = np.append(proxy_population_std_all[proxy], proxy_population_std_obs[proxy])
age_tensor = at.zeros((len(proxy_all[proxies[0]]),))
count = 0
for sub in np.arange(0, len(ages_all)):
for i in np.arange(0, ages_all[sub].shape.eval()[0]):
age_tensor = at.set_subtensor(age_tensor[count], ages_all[sub][i])
count += 1
# shared sample ages
ages = pm.Deterministic('ages', age_tensor)
# likelihood function for each proxy
proxy_idx = {}
proxy_sigma_vec = {}
proxy_population_std_vec = {}
proxy_quadrature_uncertainty = {}
if ('section' in offset_types) or ('groups' in offset_types):
offset = {}
offset_tensor = {}
if ('section' in noise_types) or ('groups' in noise_types):
noise = {}
noise_tensor = {}
for proxy in proxies:
# convert boolean vector (ignore 'exclude' and nans) to list of indices
proxy_idx[proxy] = np.where(include_idx_all[proxy])[0] #np.where(~np.isnan(proxy_all[proxy]))[0]
proxy_sigma_vec[proxy] = proxy_sigma_all[proxy][proxy_idx[proxy]]
proxy_sigma_vec[proxy][np.isnan(proxy_sigma_vec[proxy])] = proxy_sigma_default[proxy]
proxy_population_std_vec[proxy] = proxy_population_std_all[proxy][proxy_idx[proxy]]
proxy_population_std_vec[proxy][np.isnan(proxy_population_std_vec[proxy])] = 0
proxy_quadrature_uncertainty[proxy] = np.sqrt((proxy_sigma_vec[proxy])**2 + (proxy_population_std_vec[proxy])**2)
# construct final offset terms
if offset_type[proxy] != 'none':
offset_tensor[proxy] = at.zeros((len(proxy_all[proxy][proxy_idx[proxy]]),))
count = 0
# custom offset groups
if offset_type[proxy] == 'groups':
for sub in np.arange(0, len(offset_all[proxy])):
for i in np.arange(0, len(offset_all[proxy][sub])):
offset_tensor[proxy] = at.set_subtensor(offset_tensor[proxy][count], offset_all[proxy][sub][i][0])
count +=1
offset[proxy] = pm.Deterministic('offset_' + proxy, offset_tensor[proxy])
elif offset_type[proxy] == 'section':
for sub in np.arange(0, len(offset_all[proxy])):
for i in np.arange(0, offset_all[proxy][sub].shape.eval()[0]):
offset_tensor[proxy] = at.set_subtensor(offset_tensor[proxy][count], offset_all[proxy][sub][i])
count +=1
offset[proxy] = pm.Deterministic('offset_' + proxy, offset_tensor[proxy])
# construct final noise terms
if noise_type[proxy] != 'none':
noise_tensor[proxy] = at.zeros((len(proxy_all[proxy][proxy_idx[proxy]]),))
count = 0
# custom noise groups
if noise_type[proxy] == 'groups':
for sub in np.arange(0, len(noise_all[proxy])):
for i in np.arange(0, len(noise_all[proxy][sub])):
noise_tensor[proxy] = at.set_subtensor(noise_tensor[proxy][count], noise_all[proxy][sub][i][0])
count +=1
noise[proxy] = pm.Deterministic('noise_' + proxy, noise_tensor[proxy])
# per-section noise
elif noise_type[proxy] == 'section':
for sub in np.arange(0, len(noise_all[proxy])):
for i in np.arange(0, noise_all[proxy][sub].shape.eval()[0]):
noise_tensor[proxy] = at.set_subtensor(noise_tensor[proxy][count], noise_all[proxy][sub][i])
count +=1
noise[proxy] = pm.Deterministic('noise_' + proxy, noise_tensor[proxy])
if not approximate:
f = gp[proxy].prior('f_' + proxy, X=ages[proxy_idx[proxy],None],
reparameterize=True,
jitter = jitter)
if approximate:
f = gp[proxy].prior('f_' + proxy, X=ages[proxy_idx[proxy],None])
if proxy_observed:
if (offset_type[proxy] != 'none') and (noise_type[proxy] != 'none'):
proxy_pred = pm.Normal(proxy + '_pred', mu=f.flatten() + offset[proxy],
sigma = proxy_quadrature_uncertainty[proxy] + noise[proxy],
shape = proxy_all[proxy][proxy_idx[proxy]].shape,
observed=proxy_all[proxy][proxy_idx[proxy]])
# pull out the predicted proxy value for each observation (GP + offset) as a deterministic (i.e., predicted value w/out noise)
proxy_pred_mu = pm.Deterministic(proxy + '_' + '_pred_mu', f.flatten() + offset[proxy])
elif (offset_type[proxy] == 'none') and (noise_type[proxy] == 'none'):
proxy_pred = pm.Normal(proxy + '_pred', mu=f.flatten(),
sigma = proxy_quadrature_uncertainty[proxy],
shape = proxy_all[proxy][proxy_idx[proxy]].shape,
observed=proxy_all[proxy][proxy_idx[proxy]])
proxy_pred_mu = pm.Deterministic(proxy + '_' + '_pred_mu', f.flatten())
elif (offset_type[proxy] != 'none') and (noise_type[proxy] == 'none'):
proxy_pred = pm.Normal(proxy + '_pred', mu=f.flatten()+ offset[proxy],
sigma = proxy_quadrature_uncertainty[proxy],
shape = proxy_all[proxy][proxy_idx[proxy]].shape,
observed=proxy_all[proxy][proxy_idx[proxy]])
proxy_pred_mu = pm.Deterministic(proxy + '_' + '_pred_mu', f.flatten()+ offset[proxy])
elif (offset_type[proxy] == 'none') and (noise_type[proxy] != 'none'):
proxy_pred = pm.Normal(proxy + '_pred', mu=f.flatten(),
sigma = proxy_quadrature_uncertainty[proxy] + noise[proxy],
shape = proxy_all[proxy][proxy_idx[proxy]].shape,
observed=proxy_all[proxy][proxy_idx[proxy]])
proxy_pred_mu = pm.Deterministic(proxy + '_' + '_pred_mu', f.flatten())
else:
if offset_type[proxy] != 'none':
proxy_pred = pm.Normal(proxy + '_pred', mu=f.flatten() + offset[proxy],
sigma = proxy_quadrature_uncertainty[proxy] + noise[proxy],
shape = proxy_all[proxy][proxy_idx[proxy]].shape)
proxy_pred_mu = pm.Deterministic(proxy + '_' + '_pred_mu', f.flatten()+ offset[proxy])
else:
proxy_pred = pm.Normal(proxy + '_pred', mu=f.flatten(),
sigma = proxy_quadrature_uncertainty[proxy] + noise[proxy],
shape = proxy_all[proxy][proxy_idx[proxy]].shape)
proxy_pred_mu = pm.Deterministic(proxy + '_' + '_pred_mu', f.flatten())
return model, gp
[docs]
def build_prior_age_model(sample_df, ages_df, proxies = ['d13c'], **kwargs):
"""
Create a prior age model for each section. Omits the proxy signal component of the statistical model.
Parameters
----------
sample_df: pandas.DataFrame
:class:`pandas.DataFrame` containing proxy data for all sections. Load from .csv file using :py:meth:`load_data() <stratmc.data.load_data>` in :py:mod:`stratmc.data`.
ages_df: pandas.DataFrame
:class:`pandas.DataFrame` containing age constraints for all sections. Load from .csv file using :py:meth:`load_data() <stratmc.data.load_data>` in :py:mod:`stratmc.data`.
proxies: str or list(str), optional
Column or columns containing proxy data in ``sample_df``. Defaults to 'd13c'.
sections:: list(str) or numpy.array(str), optional
List of sections to include in the inference model. Defaults to all sections in ``sample_df``.
Returns
-------
prior_age_model: PyMC model
:class:`pymc.model.core.Model` object.
"""
if 'sections' in kwargs:
sections = list(kwargs['sections'])
else:
sections = list(np.unique(sample_df['section']))
if 'superposition_dict' in kwargs:
superposition_dict = kwargs['superposition_dict']
else:
superposition_dict = {}
for section in list(superposition_dict.keys()):
for older_section in superposition_dict[section]:
sections.remove(older_section)
sections.insert(0, older_section)
## instead of removing samples from dataframe if they don't have any proxy observations, mark as excluded (so the model will still keep track of age at that height)
# note - samples whose age shouldn't be tracked should simply be removed from the dataframe prior to running the inversion
# for sample_df, 'exclude' now means exclude from the likelihood calculation, but keep track of age at that height
sample_df, ages_df = clean_data(sample_df, ages_df, proxies, sections)
# ignore sections that have no observations included in the inference (samples w/ no observations were marked `Exclude? = True` in clean_data)
data_sections = list(np.unique(sample_df[~sample_df['Exclude?']]['section']))
for section in list(sections):
if section not in data_sections:
sections.remove(section)
# clean again (avoids issues w/ offset and noise groups)
sample_df, ages_df = clean_data(sample_df, ages_df, proxies, sections)
# check that for all constraints with 'shared == True', the constraint actually is used >1 time (if not, set shared = False)
for shared_age_name in ages_df[ages_df['shared?']==True]['name']:
if ages_df[(ages_df['shared?'] == True) & (ages_df['name']==shared_age_name)].shape[0] < 2:
idx = ages_df[(ages_df['shared?'] == True) & (ages_df['name']==shared_age_name)].index
ages_df.loc[idx, 'shared?'] = False
with pm.Model() as prior_age_model:
# create distribution objects for each unique shared constraint
# in this implementation, constraints should only be labeled as 'shared?' if diachronous behavior is not allowed -- constraints that are the same (e.g.,
# a fossil first appearanace date), but that may be diachronous between sections, should be set to shared = False
# create distribution objects for each unique shared constraint
# in this implementation, constraints should only be labeled as 'shared?' if diachronous behavior is not allowed -- constraints that are the same (e.g.,
# a fossil first appearanace date), but that may be diachronous between sections, should be set to shared = False
shared_constraints = {}
shared_ages = ages_df[ages_df['shared?']==True]
if len(shared_ages) > 0:
unique_shared_constraints = np.unique(shared_ages['name'])
for constraint in unique_shared_constraints:
constraint = str(constraint)
constraint_df = shared_ages[shared_ages['name']==constraint]
dist = np.unique(constraint_df['distribution_type'])
dist_age = np.unique(constraint_df['age'])
dist_age_std = np.unique(constraint_df['age_std'])
if (len(dist) > 1) or (len(dist_age) > 1) or (len(dist_age_std) > 1):
sys.exit(f"Initialization of shared age constraint {constraint} is inconsistent. Check that distribution type and parameters are the same for each section.")
dist = dist[0]
if dist == 'Normal':
shared_constraints[constraint] = pm.Normal(constraint, mu = dist_age[0], sigma = dist_age_std[0])
else:
# if not implemented, throw error
if dist not in DIST_DICT.keys():
sys.exit(f"{dist} distribution not implemented. Add to DIST_DICT or choose a different distribution.")
dist_args = {}
param_1 = constraint_df['param_1'].values[0]
param_1_name = constraint_df['param_1_name'].values[0]
param_2 = constraint_df['param_2'].values[0]
param_2_name = constraint_df['param_2_name'].values[0]
if not pd.isna(param_1):
dist_args[param_1_name] = param_1
if not pd.isna(param_2):
dist_args[param_2_name] = param_2
shared_constraints[constraint] = DIST_DICT[dist](constraint, **dist_args)
# dictionary to store sample age distributions for each section -- required for superposition between sections
section_age_dist = {}
for section in sections:
intervals = []
ages = []
section_df = sample_df[sample_df['section']==section]
# separate dataframes for intermediate detrital or intrusive constraints and depositional age constraints
section_age_df = ages_df[(ages_df['section']==section) & (~ages_df['intermediate detrital?']) & (~ages_df['intermediate intrusive?']) & (~ages_df['depositional?'])]
intermediate_detrital_section_ages_df = ages_df[(ages_df['section']==section) & (ages_df['intermediate detrital?'])]
intermediate_intrusive_section_ages_df = ages_df[(ages_df['section']==section) & (ages_df['intermediate intrusive?'])]
depositional_section_ages_df = ages_df[(ages_df['section']==section) & (ages_df['depositional?'])]
section_ages = section_age_df['age'].values
section_ages_unc = section_age_df['age_std'].values
heights = section_df['height'].values
age_heights = section_age_df['height'].values
# grab list of depositional age constraint names for current section (must match name of age in ages_df)
depositional_age_names = section_df['depositional age'].dropna().unique()
# create age constraint distributions
if len(age_heights) == 0:
print(str(section) + ' has no age constraints')
else:
ages = {}
label = str(section) +'_'
# the input ages (the means) need to be in stratigraphic superposition, but ordered transform is still used so that the posteriors cannot be out of superposition due to 2sigma uncertainty -- avoids potentially bad initialization
# biuld distributions if all age constraint priors are Gaussian and not shared
if all(section_age_df['distribution_type']=='Normal') and all(section_age_df['shared?']==False):
# for initvals, using np.sort instead of np.flip to account for scenario where means of reported ages are out of superposition
radiometric_age_flip = pm.Normal(label + 'flip_radiometric_age',
mu = np.flip(section_ages),
sigma = np.flip(section_ages_unc),
shape = section_ages.shape,
transform = tr.Ordered(),
initval = np.sort(section_ages))
radiometric_age_tensor = pm.Deterministic(label + 'radiometric_age', np.flip(radiometric_age_flip))
else:
print('Using radiometric age priors specified in ages_df for section ' + str(section))
radiometric_age = {}
age_dist_names = []
for i in np.arange(len(section_ages)):
label = str(section) + '_' + str(i) + '_'
dist = section_age_df['distribution_type'].values[i]
constraint_shared = section_age_df['shared?'].values[i]
# for shared constraints, link to existing distribution
if constraint_shared == True:
shared_constraint_name = section_age_df['name'].values[i]
radiometric_age[i] = shared_constraints[shared_constraint_name]
age_dist_names.append(shared_constraint_name)
# if the constraint is not shared, build a new distribution
else:
if dist == 'Normal':
radiometric_age[i] = pm.Normal(label + 'radiometric_age', mu = section_ages[i], sigma = section_ages_unc[i])
age_dist_names.append(label + 'radiometric_age')
else:
# if not implemented, throw error
if dist not in DIST_DICT.keys():
sys.exit(f"{dist} distribution not implemented. Add to DIST_DICT or choose a different distribution.")
dist_args = {}
param_1 = section_age_df['param_1'].values[i]
param_1_name = section_age_df['param_1_name'].values[i]
param_2 = section_age_df['param_2'].values[i]
param_2_name = section_age_df['param_2_name'].values[i]
if not pd.isna(param_1):
dist_args[param_1_name] = param_1
if not pd.isna(param_2):
dist_args[param_2_name] = param_2
radiometric_age[i] = DIST_DICT[dist](label + 'radiometric_age', **dist_args)
age_dist_names.append(label + 'radiometric_age')
# make a vector of all the radiometric ages for superposition function
radiometric_age_tensor = at.zeros((len(section_ages),))
for i in np.arange(len(section_ages)):
radiometric_age_tensor = at.set_subtensor(radiometric_age_tensor[i], radiometric_age[i])
label = str(section) +'_'
radiometric_age_tensor = pm.Deterministic(label + 'radiometric_age', radiometric_age_tensor)
superposition(radiometric_age_tensor, age_dist_names, prior_age_model, section_age_df, section)
# if there are samples below the basal age constraint, throw an error
if (heights[0] < age_heights[0]):
sys.exit(f"Section {section} does not have a basal age constraint. Add a maximum section age to ages_df.")
# if the section has no upper age constraint, throw an error
if heights[-1] >= age_heights[-1]:
sys.exit(f"Section {section} does not have an upper age constraint. Add a minimum section age to ages_df.")
# create sample age distributions for section (by interval)
for interval in np.arange(0, len(age_heights)-1).tolist():
label = str(section)+'_'+ str(interval) +'_'
above = section_df['height']>=age_heights[interval]
below = section_df['height']<age_heights[interval+1]
interval_df = section_df[above & below]
# interval_samples = interval_df[proxy].values
interval_superposition = interval_df['superposition?'].values
interval_heights = interval_df['height'].values
above = intermediate_detrital_section_ages_df['height']>age_heights[interval]
below = intermediate_detrital_section_ages_df['height']<age_heights[interval+1]
detrital_interval_df = intermediate_detrital_section_ages_df[above & below]
above = intermediate_intrusive_section_ages_df['height']>age_heights[interval]
below = intermediate_intrusive_section_ages_df['height']<age_heights[interval+1]
intrusive_interval_df = intermediate_intrusive_section_ages_df[above & below]
# if there are samples in the current interval
# if interval = 0, and the current section is in the superposition dictionary, then use the minimum age from sections that must be older as the maximum age for the current section
# note: if appropriate, make sure that the overlying age constraint is shared between the sections to avoid potential superposition issues
if len(interval_heights) > 0:
if section in list(superposition_dict.keys()):
base_age_dist = pm.math.concatenate([section_age_dist[older_section] for older_section in superposition_dict[section]]).min()
else:
base_age_dist = radiometric_age_tensor[interval]
observed_age_diff = pm.Deterministic(label + 'obs_age_diff', base_age_dist - radiometric_age_tensor[interval+1])
# sort random draws if >1 sample
if len(interval_heights) > 1:
shuffle_heights = np.unique(interval_heights[~interval_superposition])
random_sample_ages_unsorted = pm.Uniform(label + 'unsorted_random_ages', lower = 0, upper = 1, size = len(interval_heights))
# random_sample_ages_unsorted = pm.Deterministic(label + 'unsorted_random_ages_scaled', random_sample_ages_unsorted_unscaled/1e6)
# if superposition is known for all samples, sort random ages
if all(interval_superposition):
random_sample_ages = pm.Deterministic(label + 'random_ages', at.sort(random_sample_ages_unsorted))
# if there's no superposition information for any samples, skip sorting
elif (all(~interval_superposition)) and (len(shuffle_heights) == 1):
random_sample_ages = pm.Deterministic(label + 'random_ages', random_sample_ages_unsorted)
# if only some samples are missing superposition information, only sort samples w/ superposition = True
else:
sorted_idx = at.argsort(random_sample_ages_unsorted)
# create a dictionary to store lists of indices to shuffle (one list per stratigraphic horizon)
shuffle_group_idx = {}
for h in shuffle_heights:
shuffle_group_idx[h] = []
# grab indices for each group of unsorted samples
for i, h in enumerate(interval_heights):
if h in shuffle_heights:
shuffle_group_idx[h].append(i)
# sort all of the ages
random_sample_ages = at.sort(random_sample_ages_unsorted)
# unsort each group of samples without superposition information (base to top)
for i, h in enumerate(shuffle_heights):
# replace w/ another set of random draws from Uniform(0, 1), then scale between bounding samples
# note - not possible to re-shuffle the original draws (permutations not allowed in logp graph)
interval_shuffled_random_ages = pm.Uniform('shuffled_ages_' + str(section) + '_' + str(h),
lower = 0,
upper = 1,
size = len(shuffle_group_idx[h])
)
# get indices of samples below and above the shuffled range
start_shuffle_idx = np.max([0, shuffle_group_idx[h][0]-1])
stop_shuffle_idx = np.min([len(interval_heights) - 1, shuffle_group_idx[h][-1] + 1])
shuffle_base_age = random_sample_ages[start_shuffle_idx]
shuffle_top_age = random_sample_ages[stop_shuffle_idx]
# check if the starting index is also from a previously shuffled interval. if yes, reset base age to youngest sample in that group
# don't need to worry about the overlying interval, because it hasn't been reset yet
if i != 0:
start_shuffle_height = interval_heights[start_shuffle_idx]
# if the underlying sample was also shuffled, find the youngest sample in its group, and use it as the new 'base age'
if start_shuffle_height in shuffle_heights[:i]:
under_shuffle_idx = shuffle_group_idx[start_shuffle_height]
# note: because of how the scaled ages are calculated (age = base - unscaled age), the youngest sample will have the highest value in [0, 1]
shuffle_base_age = at.max(random_sample_ages[under_shuffle_idx])
# calculate (unscaled) total time spanned by the shuffled interval
# observed_shuffle_age_diff = random_sample_ages[stop_shuffle_idx] - random_sample_ages[start_shuffle_idx]
# note: ages still flipped s.t. larger values = younger
observed_shuffle_age_diff = shuffle_top_age - shuffle_base_age
# scale the shuffled [0, 1] ages s.t. the values fall in between the (sorted) values for over/underlying samples
# scaling: max - (random * observed diff)
interval_shuffled_scaled_random_ages = pm.Deterministic('shuffled_scaled_ages_' + str(section) + '_' + str(h),
shuffle_top_age - (interval_shuffled_random_ages * observed_shuffle_age_diff))
# insert unsorted random ages in tensor
random_sample_ages = at.set_subtensor(random_sample_ages[shuffle_group_idx[h]], interval_shuffled_scaled_random_ages)
# store final random age tensor in deterministic
random_sample_ages = pm.Deterministic(label + 'random_ages', random_sample_ages)
# skip sorting if interval only contains 1 sample
else:
random_sample_ages = pm.Uniform(label + 'random_ages', lower = 0, upper = 1, size = len(interval_heights))
# scaled age parameterization
scaling_factor_1 = pm.Uniform(label + 'scaling_factor_1', lower = 0, upper = 1, size = 1)
scaling_factor_2 = pm.Uniform(label + 'scaling_factor_2', lower = 0, upper = 1, size = 1)
# scaled_ages = max - random_ages * age_range * sf1 - (1 - sf2) * age_range
ages[interval] = pm.Deterministic(label + 'ages', base_age_dist - (random_sample_ages * observed_age_diff * scaling_factor_1) - (1 - scaling_factor_2) * observed_age_diff * (1 - scaling_factor_1))
intervals.append(interval)
# if there are intermediate detrital ages in the section, check if they're inside the current interval (iterate over constraints)
detrital_age_dist_names = []
detrital_age_dist_names_radio = []
if detrital_interval_df.shape[0] > 0:
# enforce detrital ages -- note that initial values will be set base --> top
for i in np.arange(detrital_interval_df.shape[0]):
# if there are overlying samples in interval, enforce maximum age
# construct DZ age prior
dist = detrital_interval_df['distribution_type'].values[i]
constraint_shared = detrital_interval_df['shared?'].values[i]
if constraint_shared == True:
shared_constraint_name = detrital_interval_df['name'].values[i]
intermediate_detrital_age = shared_constraints[shared_constraint_name]
intermediate_detrital_age_dist_name = shared_constraint_name
else:
if dist == 'Normal':
intermediate_detrital_age = pm.Normal(label + 'detrital_age_' + str(i),
mu = detrital_interval_df['age'].values[i],
sigma = detrital_interval_df['age_std'].values[i])
intermediate_detrital_age_dist_name = label + 'detrital_age_' + str(i)
else:
# if distribution not implemented, throw error
if dist not in DIST_DICT.keys():
sys.exit(f"{dist} distribution not implemented. Add to DIST_DICT or choose a different distribution.")
dist_args = {}
param_1 = detrital_interval_df['param_1'].values[i]
param_1_name = detrital_interval_df['param_1_name'].values[i]
param_2 = detrital_interval_df['param_2'].values[i]
param_2_name = detrital_interval_df['param_2_name'].values[i]
if not pd.isna(param_1):
dist_args[param_1_name] = param_1
if not pd.isna(param_2):
dist_args[param_2_name] = param_2
intermediate_detrital_age = DIST_DICT[dist](label + 'detrital_age_' + str(i), **dist_args)
intermediate_detrital_age_dist_name = label + 'detrital_age_' + str(i)
# if there are samples above the detrital age, enforce with potential
if len(interval_df[interval_df['height']>=detrital_interval_df['height'].values[i]]['height'].values)>0:
detrital_age_dist_names.append(intermediate_detrital_age_dist_name)
detrital_age_dist_names_radio.append(intermediate_detrital_age_dist_name)
# enforce detrital age with pm.Potential
intermediate_detrital_potential(intermediate_detrital_age,
intermediate_detrital_age_dist_name,
ages[interval],
interval_df['height'].values,
detrital_interval_df['height'].values[i],
section
)
# variables:
# dz age: intermediate_detrital_age
# dz age name: intermediate_detrital_age_dist_name
# sample ages (unsorted random draws U[0, 1]): random_sample_ages_unsorted
# sample age dist name: label + 'unsorted_random_ages'
# base age dist name: see above
# upper age dist name: see above
# scaling factor 1 name: label + 'scaling_factor_1'
# scaling factor 2 name: label + 'scaling_factor_2'
# if there are no samples above the current detrital constraint, just add its name to list of detritals that apply to overlying depositional age constraint
else:
detrital_age_dist_names_radio.append(intermediate_detrital_age_dist_name)
# if there are intermediate intrusive ages in section, check if they're inside this interval
intrusive_age_dist_names = []
intrusive_age_dist_names_radio = []
if intrusive_interval_df.shape[0] > 0:
# enforce intrusive ages -- note that initial values will be set base --> top
for i in np.arange(intrusive_interval_df.shape[0]):
# construct intrusive age prior
dist = intrusive_interval_df['distribution_type'].values[i]
constraint_shared = intrusive_interval_df['shared?'].values[i]
if constraint_shared == True:
shared_constraint_name = intrusive_interval_df['name'].values[i]
intermediate_intrusive_age = shared_constraints[shared_constraint_name]
intermediate_intrusive_age_dist_name = shared_constraint_name
else:
if dist == 'Normal':
intermediate_intrusive_age = pm.Normal(label + 'intrusive_age_' + str(i),
mu = intrusive_interval_df['age'].values[i],
sigma = intrusive_interval_df['age_std'].values[i])
intermediate_intrusive_age_dist_name = label + 'intrusive_age_' + str(i)
else:
# if distribution not implemented, throw error
if dist not in DIST_DICT.keys():
sys.exit(f"{dist} distribution not implemented. Add to DIST_DICT or choose a different distribution.")
dist_args = {}
param_1 = intrusive_interval_df['param_1'].values[i]
param_1_name = intrusive_interval_df['param_1_name'].values[i]
param_2 = intrusive_interval_df['param_2'].values[i]
param_2_name = intrusive_interval_df['param_2_name'].values[i]
if not pd.isna(param_1):
dist_args[param_1_name] = param_1
if not pd.isna(param_2):
dist_args[param_2_name] = param_2
intermediate_intrusive_age = DIST_DICT[dist](label + 'intrusive_age_' + str(i), **dist_args)
intermediate_intrusive_age_dist_name = label + 'intrusive_age_' + str(i)
# if there are samples below intrusive age, enforce with potential
if len(interval_df[interval_df['height']<=intrusive_interval_df['height'].values[i]]['height'].values)>0:
intrusive_age_dist_names.append(intermediate_intrusive_age_dist_name)
intrusive_age_dist_names_radio.append(intermediate_intrusive_age_dist_name)
intermediate_intrusive_potential(intermediate_intrusive_age,
intermediate_intrusive_age_dist_name,
ages[interval],
interval_df['height'].values,
intrusive_interval_df['height'].values[i],
section,
)
# if there are no samples beneath the current intrusive constraint, just add its name to list of intrusives that apply to underlying depositional age constraint
else:
intrusive_age_dist_names_radio.append(intermediate_intrusive_age_dist_name)
if (detrital_interval_df.shape[0] > 0) or (intrusive_interval_df.shape[0] > 0):
if all(section_age_df['distribution_type']=='Normal') and all(section_age_df['shared?']==False):
# THESE WILL BE UPSIDE DOWN -- inside of the intrusive potential, use this name, but flip the initial values
# with np.flip() THEN index with [interval] and [interval+1]
base_age_dist_name = str(section) +'_' + 'flip_radiometric_age'
upper_age_dist_name = str(section) +'_' + 'flip_radiometric_age'
shared_radiometric_age_dist = True
base_age_idx = interval
top_age_idx = interval + 1
else:
base_age_dist_name = age_dist_names[interval]
upper_age_dist_name = age_dist_names[interval + 1]
shared_radiometric_age_dist = False
base_age_idx = None
top_age_idx = None
if len(intrusive_age_dist_names_radio) > 0:
# enforce superposition between basal age constraint and intrusive ages
# NOTE: base_age_dist has to be flipped before using index
superposition_depositional_and_limiting_ages(prior_age_model, base_age_dist_name, [], intrusive_age_dist_names_radio, section, depositional_age_idx = base_age_idx)
if len(detrital_age_dist_names_radio) > 0:
# enforce superposition between top age constriant and detrital ages
# NOTE: upper_age_dist has to be flipped before using index inside of function
superposition_depositional_and_limiting_ages(prior_age_model, upper_age_dist_name, detrital_age_dist_names_radio, [], section, depositional_age_idx = top_age_idx)
if (len(detrital_age_dist_names) > 0) or (len(intrusive_age_dist_names) > 0):
if len(interval_heights) > 1:
age_label = 'unsorted_random_ages'
else:
age_label = 'random_ages'
get_valid_initial_ages(detrital_age_dist_names,
intrusive_age_dist_names,
base_age_dist_name,
upper_age_dist_name,
label + age_label,
interval_df['height'].values,
detrital_interval_df['height'].values,
intrusive_interval_df['height'].values,
prior_age_model,
interval,
sf1_name = label + 'scaling_factor_1',
sf2_name = label + 'scaling_factor_2',
shared_radiometric_age_dist = shared_radiometric_age_dist)
label = str(section) + '_'
# concatenate ages from all intervals in section
ages = [ages[interval] for interval in intervals]
section_age_tensor = at.zeros((len(heights),))
count = 0
for age_sub in ages:
for i in np.arange(0, age_sub.shape.eval()[0]):
section_age_tensor = at.set_subtensor(section_age_tensor[count], age_sub[i])
count += 1
section_age_dist[section] = pm.Deterministic(label + 'ages', section_age_tensor)
## for samples with depositional ages, enforce with a likelihood function
if len(depositional_age_names) > 0:
for constraint in depositional_age_names:
print(f'Adding depositional age likelihood term for section {section}: {constraint}')
age_mu = np.unique(depositional_section_ages_df[depositional_section_ages_df['name'] == constraint]['age'])
age_std = np.unique(depositional_section_ages_df[depositional_section_ages_df['name'] == constraint]['age_std'])
if (len(age_mu) > 1) or (len(age_std) > 1):
sys.exit(f"Initialization of depositional age constraint {constraint} is inconsistent. Check that the mean and standard deviation are the same for each instance in the ages DataFrame.")
elif (len(age_mu) == 0) or (len(age_std) == 0):
sys.exit(f"Depositional age constraint {constraint} not included in age constraint DataFrame for section {section}. Check that the constraint name in ages_df matches the name in sample_df, and that the constraint has not been excluded.")
else:
# grab indices of samples with the current depositional age
dep_constraint_idx = np.where(section_df['depositional age'] == constraint)[0]
# likelihood function: mean = modeled sample age, sigma = depositioanl age constraint standard deviation, observed = depositional age constraint mean
dep_age_dist = pm.Normal(str(section) + '_depositional_age_likelihood_' + constraint,
mu = section_age_dist[section][dep_constraint_idx],
sigma = list([age_std[0]]) * len(dep_constraint_idx),
observed = list([age_mu[0]]) * len(dep_constraint_idx)
)
return prior_age_model
[docs]
def superposition(age_dist, age_dist_names, model, section_age_df, section):
"""
Helper function for explicitly enforcing stratigraphic superposition (any given sample must be younger than the underlying sample) for a group of radiometric age constraints. Each constraint must have a unique name in model.
Parameters
----------
age_dist: pymc.distribtions
:class:`pytensor.tensor` containing radiometric age distributions (must be in stratigraphic order - lowest to highest).
age_dist_names: list(str)
Names of each entry in ``age_dist`` in ``model``.
model: pymc.Model
:class:`pymc.model.core.Model` object associated with the distributions in ``age_dist``.
section_age_df: pandas.DataFrame
:class:`pandas.DataFrame` containing age constraints for ``section``.
section: str
Name of the section containing the radiometric age constraints.
"""
for i in np.arange(1, age_dist.shape.eval()[0]):
# version with soft edge
slope = 100
pm.Potential("superposition_"+str(section)+'_'+str(i),
-10000 * pm.math.invlogit(slope * ((age_dist[i]) - age_dist[i-1])))
# version with hard edge - send likelihood to -infinity if age is violated
# check that lower = older (greater than) higher
# condition = at.ge(age_dist[i-1], age_dist[i])
# pm.Potential("superposition_"+str(section)+'_'+str(i), at.switch(condition, 0, -np.inf))
for i in np.arange(0, len(section_age_df['height'])-1).tolist():
lower_label = age_dist_names[i]
upper_label = age_dist_names[i + 1]
# note that model.initial_point returns transformed values, while model initial values are untransformed (?)
# so we get the initial points, and then have to apply the backward transform to 'convert' them to initial values for the untransformed distributions and check if they are in superposition
rv_var_base = model[lower_label]
base_age_initval = untransformed_initval(lower_label, model)
rv_var_upper = model[upper_label]
upper_age_initval = untransformed_initval(upper_label, model)
age_diff = base_age_initval - upper_age_initval
# if model is starting out of superposition, make upper sample younger by 1; otherwise, leave as-is:
if age_diff > 0:
model.set_initval(rv_var_base, base_age_initval)
model.set_initval(rv_var_upper, upper_age_initval)
else:
model.set_initval(rv_var_base, base_age_initval)
model.set_initval(rv_var_upper, base_age_initval - 1)
[docs]
def superposition_depositional_and_limiting_ages(model, depositional_age_name, detrital_age_dist_names, intrusive_age_dist_names, section, depositional_age_idx = None):
"""
Helper function for ensuring that depositional age constraints respect superposition with limiting age constraints. For example, a depositional age most be younger than any underlying detrital age constraints, even if their ages overlap within measurement uncertainty.
Parameters
----------
model: pymc.Model
:class:`pymc.model.core.Model` object associated with the input distributions.
depositional_age_name: str
Name of distribution for target depositional age constraint in ``model``.
detrital_age_dist_names: list(str)
Names of underlying detrital age constraint distributions in ``model``.
intrusive_age_dist_names: list(str)
Names of overlying intrusive age constraint distributions in ``model``.
section: str
Name of section containing the target age constraints.
depositional_age_idx: int
Position of ``depositional_age`` in model variable ``depositional_age_name``. Only required if ``depositional_age`` is one of multiple ages modeled using a single multidimensional distribution.
"""
if depositional_age_idx is None:
rv_var_depositional = model[depositional_age_name]
else:
rv_var_depositional_temp = at.flip(model[depositional_age_name])
rv_var_depositional = rv_var_depositional_temp[depositional_age_idx]
# grab initial value of depositional age constraint
depositional_age_initval = untransformed_initval(depositional_age_name, model)
if depositional_age_idx is not None:
depositional_age_initval = np.flip(depositional_age_initval)[depositional_age_idx]
# loop over detrital age constraints, adding potentials to enforce superposition with depositional age
for detrital_var in detrital_age_dist_names:
rv_var_detrital = model[detrital_var]
# version with soft edge
slope = 100
pm.Potential('radiometric_detrital_max_' + str(section) + '_dep_age_' + depositional_age_name + '_detrital_age_' + str(detrital_var), -10000 * pm.math.invlogit(slope * (rv_var_depositional - rv_var_detrital)))
# hard edge - send likelihood to -infinity if an age is violated
# check that depositional age is less than (younger than) detrital age
# condition = at.le(rv_var_depositional, rv_var_detrital)
# pm.Potential('radiometric_detrital_max_' + str(section) + '_dep_age_' + depositional_age_name + '_detrital_age_' + str(detrital_var), at.switch(condition, 0, -np.inf))
# check initval superposition for detrital age. if the detrital age is violated (i.e., the depositional age is older), set the detrital age initval to be 1 Myr older than the current depositional age initval
# check initval superposition for intrusive age. if the intrusive age is violated (i.e., the depositional age is younger), set the intrusive age initval to be 1 Myr younger than the current depositional age initval
detrital_age_initval = untransformed_initval(detrital_var, model)
# if depositional age constraint is older than detrital, make the detrital older (such that the depositional age is younger). note - not modifying depositional age initval, because initvals have already been set in superposition() to make sure superposition between depositional ages is respected
if depositional_age_initval > detrital_age_initval:
model.set_initval(rv_var_detrital, depositional_age_initval + 1)
# loop over intrusive age constraints, adding potentials to enforce superposition with depositional age
for intrusive_var in intrusive_age_dist_names:
rv_var_intrusive = model[intrusive_var]
# version with soft edge
slope = 100 # if using penalty of 100000, increase slope to 1000
pm.Potential('radiometric_intrusive_min_' + str(section) + '_dep_age_' + depositional_age_name + '_intrusive_age_' + str(intrusive_var), -10000 * pm.math.invlogit(slope * (rv_var_intrusive - rv_var_depositional)))
# hard edge - send likelihood to -infinity if age constraint is violated
# check that depositional age is greater than (older than) intrusive age
# condition = at.ge(rv_var_depositional, rv_var_intrusive)
# pm.Potential('radiometric_intrusive_min_' + str(section) + '_dep_age_' + depositional_age_name + '_intrusive_age_' + str(intrusive_var), at.switch(condition, 0, -np.inf))
# check initval superposition for intrusive age. if the intrusive age is violated (i.e., the depositional age is younger), set the intrusive age initval to be 1 Myr younger than the current depositional age initval
intrusive_age_initval = untransformed_initval(intrusive_var, model)
# if depositional age constraint is younger than intrusive, make the intrusive younger (such that the depositional age is older). note - not modifying depositional age initval, because initvals have already been set in superposition() to make sure superposition between depositional ages is respected
if depositional_age_initval < intrusive_age_initval:
model.set_initval(rv_var_intrusive, depositional_age_initval - 1)
# version with hard edge - sends likelihood to -infinity if constraint is violated
# condition = at.le((sample_age_dist_sorted[overlying_sample_idx]), detrital_age_dist)
# pm.Potential('detrital_max_' + str(section) + '_' + str(detrital_age_dist_name), at.switch(condition, 0, -np.inf))
# version with hard edge - send likelihoot to -infinity if age constraint is violated
# condition = at.ge(sample_age_dist_sorted[underlying_sample_idx], intrusive_age_dist)
# pm.Potential('intrusive_min_' + str(section) + '_' + str(intrusive_age_dist_name), at.switch(condition, 0, -np.inf))
[docs]
def get_valid_initial_ages(detrital_age_dist_names, intrusive_age_dist_names, maximum_age_dist_name, minimum_age_dist_name, sample_age_dist_name, sample_heights, detrital_heights, intrusive_heights, model, interval, sf1_name, sf2_name, shared_radiometric_age_dist):
"""
Helper function that resets the initial sample age values (in ``Model.initial_point()``) such that all detrital and intrusive age constraints are respected.
Parameters
----------
detrital_age_dist_names: list(str)
List of names for detrital age constraint distributions in ``model``.
intrusive_age_dist_names: list(str)
List of names for intrusive age constraint distributions in ``model``.
maximum_age_dist_name: str
Name of distribution for underlying maximum age constraint in ``model``.
minimum_age_dist_name: str
Name of distribution for overlying minimum age constraint in ``model``.
sample_age_dist_name: str
Name of sample age distribution (unsorted and unscaled) in the pymc.Model object.
sample_heights: np.array
Array of heights for samples in the current interval.
detrital_heights: list(float)
Heights of detrital age constraints.
intrusive_heights: list(float)
Heights of intrusive age constraints.
model: pymc.Model
:class:`pymc.model.core.Model` object associated with the input distributions.
section: str
Name of the current section.
interval: int
Current interval number.
sf1_name: str
Name of the distribution associated with scaling factor 1 in ``model``.
sf2_name: str
Name of the distribution associated with scaling factor 2 in ``model``.
shared_radiometric_age_dist: bool, optional
Whether the radiometric age distributions are part of a single object (versus initiated as separate distributions). Defaults to ``True``.
"""
# step 1: reset sf1 and sf2 such that it's possible for all of the age constraints to be respected (i.e., ages that are younger than the oldest detrital age, and older than the youngest intrusive age, must be inside of the 'box' of possible ages)
# get initial values for sample ages
rv_var_samples = model[sample_age_dist_name]
sample_age_initval = untransformed_initval(sample_age_dist_name, model)
# sort first -- lower likelihood that we'll have to revise initial values (and values will be sorted by model anyway)
sample_age_initval = np.sort(sample_age_initval) # smallest to largest = oldest to youngest (from base to top)
# get initial value for underlying minimum age constraint
base_age_initval = untransformed_initval(maximum_age_dist_name, model)
if shared_radiometric_age_dist:
if len(base_age_initval) > 1:
base_age_initval = np.flip(base_age_initval)[interval]
# get initial value for overlying minimum age constraint
upper_age_initval = untransformed_initval(minimum_age_dist_name, model)
if shared_radiometric_age_dist:
if len(upper_age_initval) > 1:
upper_age_initval = np.flip(upper_age_initval)[interval+1]
# set initial values for sample ages
# get initial values for scaling factors
rv_var_sf1 = model[sf1_name]
sf1_initval = untransformed_initval(sf1_name, model)
rv_var_sf2 = model[sf2_name]
sf2_initval = untransformed_initval(sf2_name, model)
# calculate the bounds of the current 'age box', given sf1 and sf2
current_max = base_age_initval - (1 - sf2_initval) * (base_age_initval - upper_age_initval) * (1 - sf1_initval)
current_min = base_age_initval - ((base_age_initval - upper_age_initval) * sf1_initval) - (1 - sf2_initval) * (base_age_initval - upper_age_initval) * (1 - sf1_initval)
scaled_dz_initvals = []
scaled_intrusive_initvals = [ ]
# iterate over detrital ages
# get initial value for detrital age constraints
for detrital_age_dist_name in detrital_age_dist_names:
dz_age_initval = untransformed_initval(detrital_age_dist_name, model)
# calculate DZ age on [0, 1] scale defined by bounds
scaled_dz_initvals.append((current_max - dz_age_initval)/(current_max - current_min))
# iterate over intrusive ages
for intrusive_age_dist_name in intrusive_age_dist_names:
intrusive_age_initval = untransformed_initval(intrusive_age_dist_name, model)
# calculate intrusive age on [0, 1] scale defined by bounds
scaled_intrusive_initvals.append((current_max - intrusive_age_initval)/(current_max - current_min))
scaled_dz_initvals = np.array(scaled_dz_initvals)
scaled_intrusive_initvals = np.array(scaled_intrusive_initvals)
# step 2: check that the 'age window' defined by the scale/shift parameters is compatible with all limiting age constraints. if not, re-draw until they do
# if any limiting age constraints are precluded by the current scale and shift parameters, reset until all age constraints can be respected:
# if a detrital age is > 1, then it's impossible for sample ages to be younger
# if an intrusive age is < 0, then it's impossible for sample age to be younger
if (any(scaled_dz_initvals > 1)) or (any(scaled_intrusive_initvals < 0)):
# randomly re-draw scale and shift parameters until age constraints can be satisfied
while (any(scaled_dz_initvals > 1)) or (any(scaled_intrusive_initvals < 0)):
sf1_initval = np.random.uniform(0, 1, 1)
sf2_initval = np.random.uniform(0, 1, 1)
current_max = base_age_initval - (1 - sf2_initval) * (base_age_initval - upper_age_initval) * (1 - sf1_initval)
current_min = base_age_initval - ((base_age_initval - upper_age_initval) * sf1_initval) - (1 - sf2_initval) * (base_age_initval - upper_age_initval) * (1 - sf1_initval)
for i, detrital_age_dist_name in enumerate(detrital_age_dist_names):
dz_age_initval = untransformed_initval(detrital_age_dist_name, model)
scaled_dz_initvals[i] = (current_max - dz_age_initval)/(current_max - current_min)
for i, intrusive_age_dist_name in enumerate(intrusive_age_dist_names):
intrusive_age_initval = untransformed_initval(intrusive_age_dist_name, model)
scaled_intrusive_initvals[i] = (current_max - intrusive_age_initval)/(current_max - current_min)
# re-calculate in case while statement was satisfied partway through recalculating initial values
for i, detrital_age_dist_name in enumerate(detrital_age_dist_names):
dz_age_initval = untransformed_initval(detrital_age_dist_name, model)
scaled_dz_initvals[i] = (current_max - dz_age_initval)/(current_max - current_min)
for i, intrusive_age_dist_name in enumerate(intrusive_age_dist_names):
intrusive_age_initval = untransformed_initval(intrusive_age_dist_name, model)
scaled_intrusive_initvals[i] = (current_max - intrusive_age_initval)/(current_max - current_min)
# set scale/shift initial values in model
model.set_initval(rv_var_sf1, sf1_initval)
model.set_initval(rv_var_sf2, sf2_initval)
# step 3: make sure the initial sample age values respect all limiting age constraints. otherwise, mcmc sampler will start in a low-probability space with a flat likelihood
# only detrital ages: set initial values, base top top (oldest to youngest)
# note - oldest to youngest should always be base top (a higher and older DZ would be useless)
if (len(detrital_age_dist_names) > 0) and (len(intrusive_age_dist_names) == 0):
for i, detrital_age_dist_name in enumerate(detrital_age_dist_names):
overlying_sample_idx = np.where(sample_heights >= detrital_heights[i])[0]
# if scaled initial value is > 0, use it to truncate the age box
# is scaled initial value is <= 0, then all samples already must be younger than the detrital age --> do nothing
if (scaled_dz_initvals[i] > 0):
# only re-draw if current (sorted) initvals don't satisfy the constraint (must be younger = larger initval)
if not all(sample_age_initval[overlying_sample_idx] > scaled_dz_initvals[i]):
scaled_age_initial_points = np.random.uniform(scaled_dz_initvals[i], 1, len(overlying_sample_idx))
sample_age_initval[overlying_sample_idx] = np.sort(scaled_age_initial_points)
# only intrusive ages: set initial values youngest to oldest (top to base)
# note - youngest to oldest should always be top to base (a lower and younger intrusive would be useless)
elif (len(detrital_age_dist_names) == 0) and (len(intrusive_age_dist_names) > 0):
for i, intrusive_age_dist_name in enumerate(np.flip(intrusive_age_dist_names)):
underlying_sample_idx = np.where(sample_heights <= np.flip(intrusive_heights)[i])[0]
# if scaled initival value is <1, use it to truncate the age box
# scaled initial value could be >=1, which means all the values in the age box already respect the constraint --> do nothing (initial values will already respect constraint)
if np.flip(scaled_intrusive_initvals)[i] < 1:
# only re-draw if current (sorted) initvals don't satisfy the constraint (must be older = smaller initval)
if not all(sample_age_initval[underlying_sample_idx] < np.flip(scaled_intrusive_initvals)[i]):
scaled_age_initial_points = np.random.uniform(0, np.flip(scaled_intrusive_initvals)[i], len(underlying_sample_idx))
sample_age_initval[underlying_sample_idx] = np.sort(scaled_age_initial_points)
# combination of detrital and an intrusive ages:
elif (len(detrital_age_dist_names) > 0) and (len(intrusive_age_dist_names) > 0):
# iterate over intrusive ages from top to base
# note - iterating over indices in reverse (top to base), so no need to flip intrusive variables inside of loop
for idx in np.flip(np.arange(len(intrusive_heights))):
# grab detrital ages below the current intrusive age
detrital_idx = np.where(detrital_heights <= intrusive_heights[idx])[0]
# set initvals in chunks bracketed by [detrital age, intrusive age], starting with the lowermost detrital age
for d_idx in detrital_idx:
# grab indices of samples in between the current intrusive and detrital constraints
sample_idx = np.where((sample_heights <= intrusive_heights[idx]) & (sample_heights >= detrital_heights[d_idx]))[0]
# if initial sample ages don't fall in between these constraints, re-draw initvals
if not (all(sample_age_initval[sample_idx] > scaled_dz_initvals[d_idx]) and all(sample_age_initval[sample_idx] < scaled_intrusive_initvals[idx])):
scaled_age_initial_points = np.random.uniform(np.max(np.concatenate([scaled_dz_initvals[d_idx], np.array([0])])),
np.min(np.concatenate([scaled_intrusive_initvals[idx], np.array([1])])),
len(sample_idx))
sample_age_initval[sample_idx] = np.sort(scaled_age_initial_points)
# if there aren't detrital ages below the current intrusive age, set initial values using only the intrusive constraint
if len(detrital_idx) == 0:
underlying_sample_idx = np.where(sample_heights <= intrusive_heights[idx])[0]
if scaled_intrusive_initvals[idx] < 1:
# only re-draw if current values don't satisfy age constraint
if not all(sample_age_initval[underlying_sample_idx] < scaled_intrusive_initvals[idx]):
scaled_age_initial_points = np.random.uniform(0, # no limit on how old
scaled_intrusive_initvals[idx], # older than intrusive
len(underlying_sample_idx))
sample_age_initval[underlying_sample_idx] = np.sort(scaled_age_initial_points)
# set all samples below the lowest detrital age to be older than the intrusive age, and also older than all the samples above the detrital age. otherwise, if initvals are actually younger than the overlying samples, then the intrusive age could be violated when the initvals are sorted
# note - doing for every intrusive age b/c we don't know where the lowest detrital age is located (ensures that samples between the lowest DZ and any underlying intrusives have valid initial ages)
if len(detrital_idx) > 0:
# grab samples below the highest intrusive age and above the lowest detrital age
# note -- not checking above the uppermost intrusive because we haven't yet checked the initvals for samples above the highest intrusive (these will be set last, and must be younger than all underlying samples)
sample_idx_above = np.where((sample_heights <= intrusive_heights[-1]) & (sample_heights >= detrital_heights[detrital_idx[0]]))[0]
# grab samples below the lowest DZ
sample_idx_below = np.where(sample_heights < detrital_heights[detrital_idx[0]])[0]
if len(sample_idx_below) > 0:
# if all of the samples below the lowest DZ aren't older than all overlying samples, re-draw
if not all(sample_age_initval[sample_idx_below] < np.min(sample_age_initval[sample_idx_above])):
# ages need to be both older than the lowermost intrusive age, and older than the oldest sample above the detrital age
scaled_age_initial_points = np.random.uniform(0, # no limit on max age
np.min(sample_age_initval[sample_idx_above]), # min age = max age (lowest initval) of overlying sample group. intrusive ages enforced as a byproduyct, since all of the underlying sample ages must be older
len(sample_idx_below))
sample_age_initval[sample_idx_below] = np.sort(scaled_age_initial_points)
# first, set everything above the highest intrusive to be younger than all underlying samples (which now have been valid initvals). then, work way up and enforce any overlying DZs
# necessary because if these initvals are older than the underlying samples, sorting would push our previously set initial values stratigraphically higher (relative to the age constraints), potentially making them invalid
# note: not a problem if we only have intrusive limiting ages
# grab samples above uppermost intrusive
overlying_sample_idx = np.where(sample_heights > np.max(intrusive_heights))[0]
# grab underlying samples -- all of these initvals were already set in previous loop
sample_below_idx = np.where(sample_heights <= np.max(intrusive_heights))[0]
if len(overlying_sample_idx) > 0:
# if all overlying samples aren't already younger (larger initvals) than the youngest underlying sample, re-draw values
if not all(sample_age_initval[overlying_sample_idx] > np.max(sample_age_initval[sample_below_idx])):
scaled_age_initial_points = np.random.uniform(np.max(sample_age_initval[sample_below_idx]), # youngest underlying sample = max age
1, # no limit on how young
len(overlying_sample_idx))
sample_age_initval[overlying_sample_idx] = np.sort(scaled_age_initial_points)
# iterate over detrital ages that are stratigraphically higher than the uppermost (youngest) intrusive age. these initial values can be set in the same way as the 'dz-only' samples (set initial values base --> top)
# note - samples in between highest intrusive and overlying DZ already taken care of by previous loop
detrital_only_idx = np.where(detrital_heights > np.max(intrusive_heights))[0]
for idx in detrital_only_idx:
overlying_sample_idx = np.where(sample_heights >= detrital_heights[idx])[0]
sample_below_idx = np.where(sample_heights < detrital_heights[idx])[0]
# if scaled initial value is > 0, use to truncate the age box
# is scaled initial value is <= 0, then all samples already must be younger than the detrital age --> do nothing
if scaled_dz_initvals[idx] > 0:
# if current initvals aren't younger (larger) than both the detrital constraint and all underlying ages, re-draw
if not all((sample_age_initval[overlying_sample_idx] > scaled_dz_initvals[idx]) & (sample_age_initval[overlying_sample_idx] > np.max(sample_age_initval[sample_below_idx]))):
# also need to be younger than all underlying samples
lower_bound = np.max(np.concatenate([scaled_dz_initvals[idx], np.max(sample_age_initval[sample_below_idx]).ravel()]))
scaled_age_initial_points = np.random.uniform(lower_bound, # younger than detrital age or youngest underlying sample (whichever is younger)
1, # no limit on how young
len(overlying_sample_idx))
sample_age_initval[overlying_sample_idx] = np.sort(scaled_age_initial_points)
model.set_initval(rv_var_samples, sample_age_initval)