Given a timetree (potentially sampled through time and not necessarily ultrametric), fit a homogenous birth-death-sampling (HBDS) model in which speciation, extinction and lineage sampling occurs at some continuous (Poissonian) rates \(\lambda\), \(\mu\) and \(\psi\), which are given as parameterized functions of time before present. Sampled lineages are kept in the pool of extant lineages at some ``retention probability'' \(\kappa\), which may also depend on time. In addition, this model can include concentrated sampling attempts at a finite set of discrete time points \(t_1,..,t_m\). ``Homogenous'' refers to the assumption that, at any given moment in time, all lineages exhibit the same speciation/extinction/sampling rates. Every HBDS model is thus defined based on the values that \(\lambda\), \(\mu\), \(\psi\) and \(\kappa\) take over time, as well as the sampling probabilities \(\psi_1,..,\psi_m\) and retention probabilities \(\kappa_1,..,\kappa_m\) during the concentrated sampling attempts; each of these parameters, in turn, is assumed to be determined by a finite set of parameters. This function estimates these parameters by maximizing the corresponding likelihood of the timetree. Special cases of this model, are sometimes known as ``birth-death-skyline plots'' in the literature (Stadler 2013). In epidemiology, these models are often used to describe the phylogenies of viral strains sampled over the course of the epidemic.
fit_hbds_model_parametric(tree,
param_values,
param_guess = NULL,
param_min = -Inf,
param_max = +Inf,
param_scale = NULL,
root_age = NULL,
oldest_age = NULL,
lambda = 0,
mu = 0,
psi = 0,
kappa = 0,
age_grid = NULL,
CSA_ages = NULL,
CSA_probs = NULL,
CSA_kappas = 0,
condition = "auto",
ODE_relative_dt = 0.001,
ODE_relative_dy = 1e-4,
CSA_age_epsilon = NULL,
Ntrials = 1,
max_start_attempts = 1,
Nthreads = 1,
max_model_runtime = NULL,
Nbootstraps = 0,
Ntrials_per_bootstrap = NULL,
fit_control = list(),
focal_param_values = NULL,
verbose = FALSE,
verbose_prefix = "")
A timetree of class "phylo", representing the time-calibrated reconstructed phylogeny of a set of extant and/or extinct species. Tips of the tree are interpreted as terminally sampled lineages, while monofurcating nodes are interpreted as non-terminally sampled lineages, i.e., lineages sampled at some past time point and with subsequently sampled descendants.
Numeric vector, specifying fixed values for a some or all model parameters. For fitted (i.e., non-fixed) parameters, use NaN
or NA
. For example, the vector c(1.5,NA,40)
specifies that the 1st and 3rd model parameters are fixed at the values 1.5 and 40, respectively, while the 2nd parameter is to be fitted. The length of this vector defines the total number of model parameters. If entries in this vector are named, the names are taken as parameter names. Names should be included if the functions lambda
, mu
, psi
, kappa
, CSA_psi
and CSA_kappa
query parameter values by name (as opposed to numeric index).
Numeric vector of size NP, specifying a first guess for the value of each model parameter. For fixed parameters, guess values are ignored. Can be NULL
only if all model parameters are fixed.
Optional numeric vector of size NP, specifying lower bounds for model parameters. If of size 1, the same lower bound is applied to all parameters. Use -Inf
to omit a lower bound for a parameter. If NULL
, no lower bounds are applied. For fixed parameters, lower bounds are ignored.
Optional numeric vector of size NP, specifying upper bounds for model parameters. If of size 1, the same upper bound is applied to all parameters. Use +Inf
to omit an upper bound for a parameter. If NULL
, no upper bounds are applied. For fixed parameters, upper bounds are ignored.
Optional numeric vector of size NP, specifying typical scales for model parameters. If of size 1, the same scale is assumed for all parameters. If NULL
, scales are determined automatically. For fixed parameters, scales are ignored. It is strongly advised to provide reasonable scales, as this facilitates the numeric optimization algorithm.
Positive numeric, specifying the age of the tree's root. Can be used to define a time offset, e.g. if the last tip was not actually sampled at the present. If NULL
, this will be calculated from the tree and it will be assumed that the last tip was sampled at the present.
Strictly positive numeric, specifying the oldest time before present (``age'') to consider when calculating the likelihood. If this is equal to or greater than the root age, then oldest_age
is taken as the stem age. If oldest_age
is less than the root age, the tree is split into multiple subtrees at that age by treating every edge crossing that age as the stem of a subtree, and each subtree is considered an independent realization of the HBDS model stemming at that age. This can be useful for avoiding points in the tree close to the root, where estimation uncertainty is generally higher. If oldest_age==NULL
, it is automatically set to the root age.
Function specifying the speciation rate at any given age (time before present) and for any given parameter values. This function must take exactly two arguments, the 1st one being a numeric vector (one or more ages) and the 2nd one being a numeric vector of size NP (parameter values), and return a numeric vector of the same size as the 1st argument with strictly positive entries. Can also be a single numeric (i.e., lambda is fixed).
Function specifying the extinction rate at any given age and for any given parameter values. This function must take exactly two arguments, the 1st one being a numeric vector (one or more ages) and the 2nd one being a numeric vector of size NP (parameter values), and return a numeric vector of the same size as the 1st argument with non-negative entries. Can also be a single numeric (i.e., mu is fixed).
Function specifying the continuous (Poissonian) lineage sampling rate at any given age and for any given parameter values. This function must take exactly two arguments, the 1st one being a numeric vector (one or more ages) and the 2nd one being a numeric vector of size NP (parameter values), and return a numeric vector of the same size as the 1st argument with non-negative entries. Can also be a single numeric (i.e., psi is fixed).
Function specifying the retention probability for continuously sampled lineages, at any given age and for any given parameter values. This function must take exactly two arguments, the 1st one being a numeric vector (one or more ages) and the 2nd one being a numeric vector of size NP (parameter values), and return a numeric vector of the same size as the 1st argument with non-negative entries. The retention probability is the probability of a sampled lineage remaining in the pool of extant lineages. Can also be a single numeric (i.e., kappa is fixed).
Numeric vector, specifying ages at which the lambda
, mu
, psi
and kappa
functionals should be evaluated. This age grid must be fine enough to capture the possible variation in \(\lambda\), \(\mu\), \(\psi\) and \(\kappa\) over time, within the permissible parameter range. Listed ages must be strictly increasing, and must cover at least the full considered age interval (from 0 to oldest_age
). Can also be NULL
or a vector of size 1, in which case \(\lambda\), \(\mu\), \(\psi\) and \(\kappa\) are assumed to be time-independent.
Optional numeric vector, listing ages (in ascending order) at which concentrated sampling attempts occurred. If NULL
, it is assumed that no concentrated sampling attempts took place and that all tips were sampled according to the continuous sampling rate psi
.
Function specifying the sampling probabilities during the various concentrated sampling attempts, depending on parameter values. Hence, for any choice of parameters, CSA_probs
must return a numeric vector of the same size as CSA_ages
. Can also be a single numeric (i.e., concentrated sampling probability is fixed).
Function specifying the retention probabilities during the various concentrated sampling attempts, depending on parameter values. Hence, for any choice of parameters, CSA_kappas
must return a numeric vector of the same size as CSA_ages
. Can also be a single numeric (i.e., retention probability during concentrated samplings is fixed).
Character, either "crown", "stem", "none" or "auto", specifying on what to condition the likelihood. If "crown", the likelihood is conditioned on the survival of the two daughter lineages branching off at the root. If "stem", the likelihood is conditioned on the survival of the stem lineage. Note that "crown" really only makes sense when oldest_age
is equal to the root age, while "stem" is recommended if oldest_age
differs from the root age. "none" is usually not recommended. If "auto", the condition is chosen according to the recommendations mentioned earlier.
Positive unitless number, specifying the default relative time step for the ordinary differential equation solvers. Typical values are 0.01-0.001.
Positive unitless number, specifying the relative difference between subsequent simulated and interpolated values, in internally used ODE solvers. Typical values are 1e-2 to 1e-5. A smaller ODE_relative_dy
increases interpolation accuracy, but also increases memory requirements and adds runtime (scaling with the tree's age span, not with Ntips).
Non-negative numeric, in units of time, specfying the age radius around a concentrated sampling attempt, within which to assume that sampling events were due to that concentrated sampling attempt. If NULL
, this is chosen automatically based on the anticipated scale of numerical rounding errors. Only relevant if concentrated sampling attempts are included.
Integer, specifying the number of independent fitting trials to perform, each starting from a random choice of model parameters. Increasing Ntrials
reduces the risk of reaching a non-global local maximum in the fitting objective.
Integer, specifying the number of times to attempt finding a valid start point (per trial) before giving up on that trial. Randomly choosen extreme start parameters may occasionally result in Inf/undefined likelihoods, so this option allows the algorithm to keep looking for valid starting points.
Integer, specifying the number of parallel threads to use for performing multiple fitting trials simultaneously. This should generally not exceed the number of available CPUs on your machine. Parallel computing is not available on the Windows platform.
Optional numeric, specifying the maximum number of seconds to allow for each evaluation of the likelihood function. Use this to abort fitting trials leading to parameter regions where the likelihood takes a long time to evaluate (these are often unlikely parameter regions).
Integer, specifying the number of parametric bootstraps to perform for estimating standard errors and confidence intervals of estimated model parameters. Set to 0 for no bootstrapping.
Integer, specifying the number of fitting trials to perform for each bootstrap sampling. If NULL
, this is set equal to max(1,Ntrials)
. Decreasing Ntrials_per_bootstrap
will reduce computation time, at the expense of potentially inflating the estimated confidence intervals; in some cases (e.g., for very large trees) this may be useful if fitting takes a long time and confidence intervals are very narrow anyway. Only relevant if Nbootstraps>0
.
Named list containing options for the nlminb
optimization routine, such as iter.max
, eval.max
or rel.tol
. For a complete list of options and default values see the documentation of nlminb
in the stats
package.
Optional numeric matrix having NP columns and an arbitrary number of rows, listing combinations of parameter values of particular interest and for which the log-likelihoods should be returned. This may be used for diagnostic purposes, e.g., to examine the shape of the likelihood function.
Logical, specifying whether to print progress reports and warnings to the screen. Note that errors always cause a return of the function (see return values success
and error
).
Character, specifying the line prefix for printing progress reports to the screen.
A list with the following elements:
Logical, indicating whether model fitting succeeded. If FALSE
, the returned list will include an additional ``error
'' element (character) providing a description of the error; in that case all other return variables may be undefined.
The maximized fitting objective. Currently, only maximum-likelihood estimation is implemented, and hence this will always be the maximized log-likelihood.
The name of the objective that was maximized during fitting. Currently, only maximum-likelihood estimation is implemented, and hence this will always be ``loglikelihood''.
The log-likelihood of the fitted model for the given timetree.
Numeric vector of size NP (number of model parameters), listing all fitted or fixed model parameters in their standard order (see details above). If param_names
was provided, elements in fitted_params
will be named.
Numeric vector of size NP, listing guessed or fixed values for all model parameters in their standard order. If param_names
was provided, elements in param_guess
will be named.
The loglikelihood of the data for the initial parameter guess (param_guess
).
A numeric vector of the same size as nrow(focal_param_values)
, listing loglikelihoods for each of the focal parameter conbinations listed in focal_loglikelihoods
.
Integer, number of fitted (i.e., non-fixed) model parameters.
Number of data points used for fitting, i.e., the number of sampling and branching events that occurred between ages 0 and oldest_age
.
The Akaike Information Criterion for the fitted model, defined as \(2k-2\log(L)\), where \(k\) is the number of fitted parameters and \(L\) is the maximized likelihood.
The Bayesian information criterion for the fitted model, defined as \(\log(n)k-2\log(L)\), where \(k\) is the number of fitted parameters, \(n\) is the number of data points (Ndata
), and \(L\) is the maximized likelihood.
Logical, specifying whether the maximum likelihood was reached after convergence of the optimization algorithm. Note that in some cases the maximum likelihood may have been achieved by an optimization path that did not yet converge (in which case it's advisable to increase iter.max
and/or eval.max
).
Integer, specifying the number of iterations performed during the optimization path that yielded the maximum likelihood.
Integer, specifying the number of likelihood evaluations performed during the optimization path that yielded the maximum likelihood.
Numeric vector of size Ntrials
, listing the initial objective values (e.g., loglikelihoods) for each fitting trial, i.e. at the start parameter values.
Numeric vector of size Ntrials
, listing the final maximized objective values (e.g., loglikelihoods) for each fitting trial.
Integer vector of size Ntrials
, listing the number of start attempts for each fitting trial, until a starting point with valid likelihood was found.
Integer vector of size Ntrials
, listing the number of iterations needed for each fitting trial.
Integer vector of size Ntrials
, listing the number of likelihood evaluations needed for each fitting trial.
Numeric vector of size NP, estimated standard error of the parameters, based on parametric bootstrapping. Only returned if Nbootstraps>0
.
Numeric vector of size NP, median the estimated parameters across parametric bootstraps. Only returned if Nbootstraps>0
.
Numeric vector of size NP, lower bound of the 50% confidence interval (25-75% percentile) for the parameters, based on parametric bootstrapping. Only returned if Nbootstraps>0
.
Numeric vector of size NP, upper bound of the 50% confidence interval for the parameters, based on parametric bootstrapping. Only returned if Nbootstraps>0
.
Numeric vector of size NP, lower bound of the 95% confidence interval (2.5-97.5% percentile) for the parameters, based on parametric bootstrapping. Only returned if Nbootstraps>0
.
Numeric vector of size NP, upper bound of the 95% confidence interval for the parameters, based on parametric bootstrapping. Only returned if Nbootstraps>0
.
Numeric between 0 and 1, estimated consistency of the data with the fitted model. See the documentation of fit_sbm_const
for an explanation.
This function is designed to estimate a finite set of scalar parameters (\(p_1,..,p_n\in\R\)) that determine the speciation rate \(\lambda\), the extinction rate \(\mu\), the sampling rate \(\psi\), the retention rate \(\kappa\), the concentrated sampling probabilities \(\psi_1,..,\psi_m\) and the concentrated retention probabilities \(\kappa_1,..,\kappa_m\), by maximizing the likelihood of observing a given timetree under the HBDS model. Note that the ages (times before present) of the concentrated sampling attempts are assumed to be known and are not fitted.
It is generally advised to provide as much information to the function fit_hbds_model_parametric
as possible, including reasonable lower and upper bounds (param_min
and param_max
), a reasonable parameter guess (param_guess
) and reasonable parameter scales param_scale
. If some model parameters can vary over multiple orders of magnitude, it is advised to transform them so that they vary across fewer orders of magnitude (e.g., via log-transformation). It is also important that the age_grid
is sufficiently fine to capture the variation of \(\lambda\), \(\mu\), \(\psi\) and \(\kappa\) over time, since the likelihood is calculated under the assumption that these functions vary linearly between grid points.
Note that in this function age always refers to time before present, i.e., present day age is 0 and age increases towards the root. The functions lambda
, mu
, psi
and kappa
should be functions of age, not forward time.
T. Stadler, D. Kuehnert, S. Bonhoeffer, A. J. Drummond (2013). Birth-death skyline plot reveals temporal changes of epidemic spread in HIV and hepatitis C virus (HCV). PNAS. 110:228-233.
# NOT RUN {
# Generate a random tree with exponentially varying lambda & mu and constant psi
# assume that all sampled lineages are removed from the pool (i.e. kappa=0)
time_grid = seq(from=0, to=100, by=0.01)
root_age = 5
tree = generate_tree_hbds(max_time = root_age,
time_grid = time_grid,
lambda = 2*exp(0.1*time_grid),
mu = 0.1*exp(0.09*time_grid),
psi = 0.1,
kappa = 0)$tree
cat(sprintf("Tree has %d tips\n",length(tree$tip.label)))
# Define a parametric HBDS model, with exponentially varying lambda & mu
# Assume that the sampling rate is constant but unknown
# The model thus has 5 parameters: lambda0, mu0, alpha, beta, psi
lambda_function = function(ages,params){
return(params['lambda0']*exp(-params['alpha']*ages));
}
mu_function = function(ages,params){
return(params['mu0']*exp(-params['beta']*ages));
}
psi_function = function(ages,params){
return(rep(params['psi'],length(ages)))
}
# Define an age grid on which lambda_function & mu_function shall be evaluated
# Should be sufficiently fine to capture the variation in lambda & mu
age_grid = seq(from=0,to=100,by=0.01)
# Perform fitting
cat(sprintf("Fitting model to tree..\n"))
fit = fit_hbds_model_parametric(tree,
param_values = c(lambda0=NA, mu0=NA, alpha=NA, beta=NA, psi=NA),
param_guess = c(1,1,0,0,0.5),
param_min = c(0,0,-1,-1,0),
param_max = c(10,10,1,1,10),
param_scale = 1, # all params are in the order of 1
lambda = lambda_function,
mu = mu_function,
psi = psi_function,
kappa = 0,
age_grid = age_grid,
Ntrials = 4, # perform 4 fitting trials
Nthreads = 2) # use 2 CPUs
if(!fit$success){
cat(sprintf("ERROR: Fitting failed: %s\n",fit$error))
}else{
cat(sprintf("Fitting succeeded:\nLoglikelihood=%g\n",fit$loglikelihood))
# print fitted parameters
print(fit$param_fitted)
}
# }
Run the code above in your browser using DataLab