-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathREADME.Rmd
388 lines (257 loc) · 13.7 KB
/
README.Rmd
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
---
output: github_document
---
<!-- README.md is generated from README.Rmd. Please edit that file -->
```{r, echo = FALSE}
knitr::opts_chunk$set(
collapse = TRUE,
comment = "#>",
fig.path = "man/figures/README-"
)
```
```{r,echo=FALSE}
library(knitr)
```
# mapbayr <img align="right" src = "inst/logo.png" width="135px">
<!-- badges: start -->
[![CRAN status](https://www.r-pkg.org/badges/version-last-release/mapbayr)](https://CRAN.R-project.org/package=mapbayr)
[![](https://img.shields.io/badge/Citation-CPT:PSP-blue.svg)](https://pubmed.ncbi.nlm.nih.gov/34342170/)
<!-- badges: end -->
mapbayr is a free and open source package for *maximum a posteriori* bayesian estimation of PK parameters in R. Thanks to a single function, `mapbayest()`, you can estimate individual PK parameters from:
* a population PK model (coded in [mrgsolve](https://github.com/metrumresearchgroup/mrgsolve)),
* a data set with concentrations (NM-TRAN format).
It was designed to be easily wrapped in [shiny apps](https://github.com/FelicienLL/mapbayr-shiny) in order to ease model-based Therapeutic Drug Monitoring, also referred to as Model-Informed Prediction Dosing (MIPD).
## Installation
mapbayr is available on [CRAN](https://CRAN.R-project.org/package=mapbayr). You can install the development version from github by executing the following code in R console.
```{r, eval = F}
install.packages("devtools")
devtools::install_github("FelicienLL/mapbayr")
```
mapbayr relies on [mrgsolve](https://github.com/metrumresearchgroup/mrgsolve) for model implementation and ordinary differential equation solving which requires C++ compilers. If you are a Windows user, you would probably need to install [Rtools](https://cran.r-project.org/bin/windows/Rtools/). Please refer to the [installation guide of mrgsolve](https://github.com/metrumresearchgroup/mrgsolve/wiki/mrgsolve-Installation) for additional information.
## Example
```{r,echo = T, results='hide', message=FALSE}
library(mapbayr)
library(mrgsolve)
```
```{r message=FALSE, include=FALSE, results='hide'}
devtools::load_all()
```
#### 1) Properly code you model
```{r, echo = T, results='hide', message=FALSE}
code <- "
$PARAM @annotated
TVCL: 0.9 : Clearance
TVV1: 10.0 : Central volume
V2 : 10.0 : Peripheral volume of distribution
Q : 1.0 : Intercompartmental clearance
ETA1: 0 : Clearance (L/h)
ETA2: 0 : Central volume (L)
$PARAM @annotated @covariates
BW : 70 : Body weight (kg)
$OMEGA 0.3 0.3
$SIGMA
0.05 // proportional
0.1 // additive
$CMT @annotated
CENT : Central compartment (mg/L)[ADM, OBS]
PERIPH: Peripheral compartment ()
$TABLE
double DV = (CENT/V1) *(1 + EPS(1)) + EPS(2);
$MAIN
double CL = TVCL * exp(ETA1 + ETA(1)) * pow(BW / 70, 1.2) ;
double V1 = TVV1 * exp(ETA2 + ETA(2)) ;
double K12 = Q / V1 ;
double K21 = Q / V2 ;
double K10 = CL / V1 ;
$ODE
dxdt_CENT = K21 * PERIPH - (K10 + K12) * CENT ;
dxdt_PERIPH = K12 * CENT - K21 * PERIPH ;
$CAPTURE DV CL
"
my_model <- mcode("Example_model", code)
```
#### 2) Bring your dataset
```{r, echo = T, message=FALSE}
my_data <- data.frame(ID = 1, time = c(0,6,15,24), evid = c(1, rep(0,3)), cmt = 1, amt = c(100, rep(0,3)),
rate = c(20, rep(0,3)), DV = c(NA, 3.9, 1.1, 2), mdv = c(1,0,0,1), BW = 90)
my_data
```
#### 3) And estimate !
```{r, message=FALSE}
my_est <- mapbayest(my_model, data = my_data)
```
As building dataset into a NM-TRAN format can be painful, you can use pipe-friendly `obs_rows()`, `adm_rows()` and `add_covariates()` functions in order to pass administration and observation information, and perform the estimation subsequently.
```{r, eval=FALSE}
my_est <- my_model %>%
adm_rows(time = 0, amt = 100, rate = 20) %>%
obs_rows(time = 6, DV = 3.9) %>%
obs_rows(time = 20, DV = 1.1) %>%
obs_rows(time = 24, DV = 2, mdv = 1) %>%
add_covariates(BW = 90) %>%
mapbayest()
```
#### 4) Then, use the estimations
The results are returned in a single object ("mapbayests" S3 class) which includes input (model and data), output (etas and tables) and internal arguments passed to the internal algorithm (useful for debugging). Additional methods are provided to ease visualization and computation of a posteriori outcomes of interest.
```{r, message=FALSE}
print(my_est)
```
```{r, plot1, fig.height=4, message=FALSE}
plot(my_est)
```
```{r, plot2, fig.height=2.5, message=FALSE, warning=FALSE}
hist(my_est)
```
```{r, plot3, message = FALSE}
# Easily extract a posteriori parameter values to compute outcomes of interest
get_eta(my_est)
get_param(my_est, "CL")
# The `use_posterior()` functions updates the model object with posterior values and covariates to simulate like with a regular mrgsolve model
my_est %>%
use_posterior() %>%
data_set(expand.ev(amt = c(50, 100, 200, 500), dur = c(5, 24)) %>% mutate(rate = amt/dur)) %>%
carry_out(dur) %>%
mrgsim() %>%
plot(DV~time|factor(dur), scales = "same")
```
## Development
mapbayr is under development. Your feedback for additional feature requests or bug reporting is welcome. Contact us through the [issue tracker](https://github.com/FelicienLL/mapbayr/issues).
## Features
mapbayr is a generalization of the "MAP Bayes estimation" tutorial available on the [mrgsolve blog](https://mrgsolve.org/blog/posts/map_bayes.html). Additional features are:
* a unique function to perform the estimation: `mapbayest()`.
* accepts a large variety of structural models thanks to the flexibility of mrgsolve
* flexibility with random effects on parameters, accepting both inter-individual and inter-occasion variability.
* additive, proportional, mixed or exponential (without prior log-transformation of data) residual error models.
* estimate from both parent drug and metabolite simultaneously.
* fit multiple patients stored in a single dataset.
* functions to easily pass administration and observation information, as well as plot methods to visualize predictions and parameter distribution.
* a single output object to ease post-processing, depending on the purpose of the estimation.
* several optimization algorithm available, such as "L-BFGS-B" (the default) or "newuoa".
* handling data below the limit of quantification.
* estimate only a subset of ETAs defined in the model.
* flatten priors to favor observed data.
## Performance
Reliability of parameter estimation against NONMEM was assessed for a wide variety of models and data. The results of this validation study were published in [CPT:Pharmacometrics & System Pharmacology](https://pubmed.ncbi.nlm.nih.gov/34342170/), and materials are available in [a dedicated repository](https://github.com/FelicienLL/mapbayr-CPTPSP-2021). If you observe some discrepancies between mapbayr and NONMEM on your own model and data, feel free to contact us through the [issue tracker](https://github.com/FelicienLL/mapbayr/issues).
## *mrgsolve* model specification
mapbayr contains a library of example model files (.cpp), accessible with `exmodel()`. You are invited to perform MAP-Bayesian estimation with your own models. These model files should be slightly modified in order to be "read" by mapbayr with the subsequent specifications:
### 1. `$PARAM` block
#### 1.1 ETA specifications
* Mandatory:
+ Add as many ETA as there are parameters to estimate (i.e. the length of the OMEGA matrix diagonal).
+ Name them as ETAn (n being the N° of ETA).
+ Set 0 as default value.
* Strongly recommended:
+ Provide a description as a plain text
```{c, eval = F}
$PARAM @annotated
ETA1 : 0 : CL (L/h)
ETA2 : 0 : VC (L)
ETA3 : 0 : F ()
//do not write ETA(1)
//do not write iETA
```
#### 1.2 Covariates
* Mandatory:
+ Use a `@covariates` tag to record covariates in the `$PARAM` block. Otherwise, you will not be allowed to pass a dataset with covariates columns.
+ Set the reference value.
* Strongly recommended
+ Provide a description as a plain text
+ Provide units in parentheses (or a description of 0/1 coding for categorical covariates)
```{c, eval = F}
$PARAM @annotated @covariates
BW : 70 : Body weight (kg)
SEX : 0 : Sex (0=Male, 1=Female)
```
### 2. `$CMT` block
* Strongly recommended...
... yet **mandatory** if you have multiple types of DV, i.e. parent drug + metabolite:
+ A `@annotated` tag must be used to record compartments.
+ Write OBS in brackets to define the observation compartment(s). Also used by `obs_rows()` to fill the 'cmt' column in your dataset.
+ Write ADM in brackets to define "default" administration compartment(s). This information is not used for optimization process and the `mapbayest()` function. The information is mandatory if you use `adm_rows()` to build your dataset in order to automatically set the value of the 'cmt' column. Especially useful if you use a model with an absorption from several depot compartment requiring to duplicate administrations lines in the data set.
```{c, eval = F}
//example: model with dual zero and first order absorption in compartment 1 & 2, respectively, and observation of parent drug + metabolite
$CMT @annotated
DEPOT: Depot [ADM]
CENT_PAR: examplinib central [ADM, OBS]
PERIPH : examplinib peripheral
CENT_MET : methylexamplinib central [OBS]
```
### 3. `$OMEGA` block
* Mandatory:
+ The length of the omega matrix must be the same as the number of ETAn provided in `$PARAM`.
+ The order of the omega values must correspond to the order of the ETAs provided in `$PARAM`. This cannot be checked by mapbayr !
```{c, eval = F}
$OMEGA
0.123 0.456 0.789
$OMEGA @block
0.111
0.222 0.333
// reminder: omega values can be recorded in multiple $OMEGA blocks
```
### 4. `$SIGMA` block
The definition of the `$SIGMA` block may not be as straightforward as other blocks, but we tried to keep it as simple as possible. Keep in mind that mapbayr always expect a **pair of sigma values** for each type of dependent variable: the **first** value for proportional error, the **second** for additive.
Two situations can be distinguished:
1. You only have one type of concentration to fit, and you did not use the [OBS] assignment in `$CMT`.
Simply write **one pair** of sigma values to describe proportional and additive error on your concentrations. This error model will be automatically applied to the compartment where observations were recorded in your dataset (i.e. value of CMT when MDV = 0).
```{c, eval = F}
$SIGMA 0.111 0 // proportional error
```
```{c, eval = F}
$SIGMA 0 0.222 // (log) additive error
```
```{c, eval = F}
$SIGMA 0.333 0.444 // mixed error
```
2. You have multiple types of DV (parent and metabolite), and/or you used the [OBS] assignment in `$CMT`.
Write as many **pairs of sigma values** as there are compartments assigned with [OBS] in `$CMT`. The order of the pair must respect the order in which compartments were assigned. To put it more clearly, the sigma matrix will be interpreted as such whatever the model :
| N° in the SIGMA matrix diagonal | Associated error |
|:-------------------------------:|:----------------:|
| 1 | Proportional on concentrations in the 1st cmt with [OBS] |
| 2 | Additive on concentrations in the 1st cmt with [OBS] |
| 3 | Proportional on concentrations in the 2nd cmt with [OBS] |
| 4 | Additive on concentrations in the 2nd cmt with [OBS] |
```{c, eval = F}
//example: correlated proportional error between parent and metabolite
$SIGMA @block
0.050 // proportional error on parent drug
0.000 0.000 // additive error on parent drug
0.100 0.000 0.200 // proportional error on metabolite
0.000 0.000 0.000 0.000 // additive error on metabolite
// reminder: sigma values can be recorded in multiple $SIGMA blocks
```
### 6. `$TABLE` block or `$ERROR` block
* Mandatory:
+ Refer the concentration variable to fit as `DV`. Mind the code, especially if concentrations are observed in multiple compartments.
+ Express log-additive error models as exponential. This way, concentrations will automatically be log-transformed during the optimization process, with no necessity to prior log-transform your concentration.
```{c, eval = F}
$TABLE
double DV = (CENTRAL / VC) * exp(EPS(2)) ;
```
+ For fitting parent drug and metabolite simultaneously, refer to them as PAR and MET, and define DV accordingly (only DV will be used during the optimization process, but PAR and MET variables are mandatory for post-processing internal functions)
```{c, eval = F}
$TABLE
double PAR = (CENT_PAR / V) * (1 + EPS(1)) ;
double MET = (CENT_MET / V) * (1 + EPS(3)) ;
double DV = PAR ;
if(self.cmt == 4) DV = MET ;
// reminder: use "self.cmt" to internaly refer to a compartment in a mrgsolve model code.
```
Note that mapbayr does not strictly rely on this `$ERROR` block to define the residual error internally and compute the objective function value, but on information passed in the `$SIGMA` block. However, we strongly advise you to properly code your `$ERROR` block with `EPS(1)`, `EPS(2)` etc..., if only to use your code as a regular mrgsolve model code and simulate random effects.
### 7. `$MAIN` block
* Mandatory:
+ Double every expression containing ETA information, with ETAn (will be used for optimization of parameters) and ETA(n) (generated for simulations with random effects like a "regular" mrgsolve model)
+ Mind the attribution to the good ETAn and ETA(n) as respect to the information you provided in `$PARAM` and `$OMEGA`. This cannot be checked by mapbayr !
```{c, eval = F}
$PK
double CL = TVCL * exp(ETA1 + ETA(1)) ;
```
### 8. `$CAPTURE` block
* Mandatory:
+ DV must be captured
+ For models with parent + metabolite, PAR and MET must be captured too.
+ Do not capture variables called IPRED and PRED (they will be returned by `mapbayest()` anyway)
+ Do not capture any ETAn (ETA1, ETA2 etc...) (they will be returned by `mapbayest()` anyway)
* Strongly recommended:
+ Capture a posteriori values of parameters you are interested in (e.g. CL)
```{c, eval = F}
$CAPTURE DV PAR MET CL
```