Estimating Parameters using SAEM
PumasEMModel
s can optionally be fitted using SAEM
. Here is an example PumasEMModel
definition:
using Pumas
using PumasUtilities
using Random
covariate_saem_cov = @emmodel begin
@random begin
CL ~ 1 + logwt | LogNormal
v ~ 1 | LogNormal
end
@covariance 2
@covariates wt
@pre begin
Vc = wt * v
end
@dynamics Central1
@post begin
cp = Central / Vc
end
@error begin
dv ~ ProportionalNormal(cp)
end
end
PumasEMModel
Parameters with random effects:
CL ~ (1, :logwt) | LogNormal
v ~ (1,) | LogNormal
Covariates: wt
Pre-dynamical variables: Vc
Dynamical system variables: Central
Post-dynamical variables: cp
See the documentation on the @emmodel
macro interface for an explanation of the syntax. To fit this model, we'll simulate data using simobs
.
sim_params_covariate_cov = (;
CL = (4.0, 0.75),
v = 70.0,
Ω = (Pumas.@SMatrix([0.1 0.05; 0.05 0.1]),),
σ = ((0.2,),),
)
obstimes = 0.0:100.0
dose = DosageRegimen(1_000; addl = 2, ii = 24)
function choose_covariates()
wt = (55 + 25rand()) / 70
return (; wt, logwt = log(wt))
end
pop = [Subject(; id = i, events = dose, covariates = choose_covariates()) for i = 1:72]
sims = simobs(
covariate_saem_cov,
pop,
sim_params_covariate_cov;
obstimes,
ensemblealg = EnsembleSerial(),
)
reread_df = DataFrame(sims);
pop_covariate = read_pumas(reread_df; observations = [:dv], covariates = [:logwt, :wt])
Population
Subjects: 72
Covariates: logwt, wt
Observations: dv
When specifying init
s, it is not necessary to specify any variance parameters for the random effects or error models.
init_covariate = (; CL = (2.0, 2 / 3), v = 50.0)
(CL = (2.0, 0.6666666666666666),
v = 50.0,)
If unspecified, they will be initialized to 1.0
or the identity matrix for covariances. SAEM
's stochastic exploration phase is most effective when these variance parameters are much larger than the true variances. Thus, if the true variances are believed to be around 1.0
or larger, it is recommended to specify larger initial values manually.
It is possible to pass a vector of random number generates as an argument to fit
. If so, the fit will use one thread per RNG
. By specifying the seeds of each RNG
, SAEM()
can be fully reproducible:
rngv = [MersenneTwister(1941964947i + 1) for i ∈ 1:Threads.nthreads()];
fit_covariate_cov1 = fit(
covariate_saem_cov,
pop_covariate,
init_covariate,
SAEM();
ensemblealg = EnsembleThreads(),
rng = rngv,
)
rngv = [MersenneTwister(1941964947i + 1) for i ∈ 1:Threads.nthreads()];
fit_covariate_cov2 = fit(
covariate_saem_cov,
pop_covariate,
init_covariate,
SAEM();
ensemblealg = EnsembleThreads(),
rng = rngv,
)
coef(fit_covariate_cov1) == coef(fit_covariate_cov2) # true
true
One can also specify the number of iterations for each of the three phases (rapid exploration, convergence, smoothing):
fit_covariate_cov =
fit(covariate_saem_cov, pop_covariate, init_covariate, SAEM(; iters = (1000, 500, 500)))
FittedPumasEMModel
Likelihood approximation: SAEM
Dynamical system type: Closed form
Log-likelihood value: -12498.5
Number of subjects: 72
Number of parameters: Fixed Optimized
0 7
Observation records: Active Missing
dv: 7272 0
Total: 7272 0
-------------------
Estimate
-------------------
CL₁ 3.6303
CL₂ 0.88717
v 66.696
Ω₁,₁ 0.098272
Ω₂,₁ 0.057117
Ω₂,₂ 0.10378
σ 0.19801
-------------------
The results of a fit can be analyzed normally:
infer_cov = infer(fit_covariate_cov)
Asymptotic inference results using sandwich estimator
Likelihood approximation: SAEM
Dynamical system type: Closed form
Log-likelihood value: -12498.5
Number of subjects: 72
Number of parameters: Fixed Optimized
0 7
Observation records: Active Missing
dv: 7272 0
Total: 7272 0
------------------------------------------------------------------
Estimate SE 95.0% C.I.
------------------------------------------------------------------
CL_base 3.6303 0.13783 [ 3.3601 ; 3.9004 ]
CL_logwt 0.88717 0.27262 [ 0.35285; 1.4215 ]
v_base 66.696 2.5462 [61.705 ; 71.686 ]
ω_1₁,₁ 0.31348 0.026764 [ 0.26103; 0.36594]
ω_1₂,₁ 0.1822 0.039445 [ 0.10489; 0.25951]
ω_1₂,₂ 0.26568 0.020637 [ 0.22523; 0.30613]
σ_0 0.19801 0.0018806 [ 0.19432; 0.20169]
------------------------------------------------------------------
coeftable(infer_cov)
Row | parameter | estimate | se | relative_se | ci_lower | ci_upper |
---|---|---|---|---|---|---|
String | Float64 | Float64 | Float64 | Float64 | Float64 | |
1 | CL_base | 3.63028 | 0.137825 | 0.0379654 | 3.36015 | 3.90041 |
2 | CL_logwt | 0.887173 | 0.27262 | 0.30729 | 0.352849 | 1.4215 |
3 | v_base | 66.6959 | 2.54623 | 0.0381767 | 61.7054 | 71.6864 |
4 | ω_1₁,₁ | 0.313484 | 0.0267636 | 0.0853749 | 0.261028 | 0.365939 |
5 | ω_1₂,₁ | 0.182201 | 0.0394447 | 0.21649 | 0.104891 | 0.259511 |
6 | ω_1₂,₂ | 0.26568 | 0.0206373 | 0.0776773 | 0.225231 | 0.306128 |
7 | σ_0 | 0.198007 | 0.00188057 | 0.0094975 | 0.194321 | 0.201693 |
inspect_cov = inspect(fit_covariate_cov)
[ Info: Calculating predictions.
[ Info: Calculating weighted residuals.
[ Info: Calculating empirical bayes.
[ Info: Evaluating individual parameters.
[ Info: Evaluating dose control parameters.
[ Info: Done.