--- title: 'Titration/Rule-Based Simulations' output: rmarkdown::html_vignette vignette: > %\VignetteIndexEntry{Titration/Rule-Based Simulations} %\VignetteEngine{knitr::rmarkdown} %\VignetteEncoding{UTF-8} --- ```{r setup, include=FALSE, eval=TRUE} knitr::opts_chunk$set(echo = TRUE, message=FALSE, eval=FALSE) require(ubiquity) require(deSolve) require(ggplot2) require(foreach) require(doParallel) require(rhandsontable) # The presim variable will contain presimualted data when eval is set to true presim_loaded = FALSE ``` ```{r echo=FALSE, results=FALSE} presim= list() if(file.exists("Titration_presim.RData")){ file.remove("Titration_presim.RData") } ``` ```{r echo=FALSE, results=FALSE, eval=TRUE} if(file.exists("Titration_presim.RData")){ load("Titration_presim.RData") presim_loaded = TRUE } ``` ## Introduction In the Simulation vignette several examples are provided using fixed dosing regimens. This works well for compounds that are relatively safe, the PK is consistent over time, and for which the drug concentrations required to achieve efficacy are consistent. In fact these restrictions apply to many different compounds. However, there are many scenarios where it is necessary to adjust dosing based on periodic clinical visits: compounds with a narrow therapeutic index, disease progression which alters the drug PK or requires increases in the dose amount, etc. While it is possible to do this manually with the current simulation tools provided, this section outlines a generalized framework to make this easier for the users. The [workshop](https://ubiquity-pkpd.s3-us-west-1.amazonaws.com/files/ubiquity_workshop.zip) provides several examples of running titration or rule-based simulations. To make a copy of these scripts and other supporting files in the current working directory run the following: ```{r eval=FALSE} library(ubiquity) fr = workshop_fetch(section="Titration", overwrite=TRUE) ``` This should provide you with the following scripts: * ``analysis_repeat_dosing.r`` - Implement fixed bolus dosing using the titration framework * ``analysis_repeat_infusion.r`` - Implement fixed infusion rates using the titration framework * ``analysis_visit_dosing_titration.r`` - Alter dosing based on predictions at specified time points (individual) * ``analysis_visit_dosing_titration_stochastic.r`` - Alter dosing based on predictions at specified time points (population) * ``analysis_state_reset.r`` - Combine multiple rules to dose the drug and perform a state reset Titration or rule-based simulations are run using the following functions: * `system_new_tt_rule()` - Defines a new rule and the times when it is evaluated * `system_set_tt_cond()` - Adds a condition to a rule and an associated action to perform when that condition is true * `run_simulation_titrate()` - Runs an individual simulation using titration rules ## Repeated Bolus Dosing (``analysis_repeat_dosing.r``) This example shows how to implement a dosing regimen using both the fixed-dosing methodologies and the titration functions above. This is meant as a means to bridge these two concepts. First we use the same model from the Simulation vignette ([Davda etal. mAbs, 6(4), 1094-1102](https://doi.org/10.4161/mabs.29095)), and simulate five doses of 500 mg IV given every two weeks: ```{r results="hide", message=FALSE, echo=FALSE} library(ubiquity) system_new(file_name="system.txt", system_file="mab_pk", overwrite = TRUE) ``` ```{r results="hide", message=FALSE, warning=FALSE} cfg = build_system() cfg = system_select_set(cfg, "default") parameters = system_fetch_parameters(cfg) cfg=system_set_option(cfg,group = "simulation", option = "solver", value = "lsoda") cfg=system_set_option(cfg, group = "simulation", option = "output_times", seq(0,10*7,1)) cfg = system_zero_inputs(cfg) cfg = system_set_bolus(cfg, state = "Cc", times = c(0, 14, 28, 42, 56), values = c(500, 500, 500, 500, 500)) som_fix = run_simulation_ubiquity(parameters, cfg) ``` To use rule-based control of the simulation we have to enable the titration option: ```{r results="hide", message=FALSE} cfg=system_set_option(cfg, group = "titration", option = "titrate", value = TRUE) ``` Next we need to define a rule using `system_new_tt_rule()`. A rule contains a set of times (and the timescale) where we want to evaluate that rule. Here we create a rule called ``"ivdose"`` and we evaluate this rule at weeks 0, 2, 4, 6 and 8. Any timescale defined using ```` in the system file can be used: ```{r results="hide", message=FALSE} cfg=system_new_tt_rule(cfg, name = "ivdose", times = c(0, 2, 4, 6, 8), timescale = "weeks") ``` Once a rule has been created you can attach conditions to that rule using `system_set_tt_cond()`. The name passed to this function is the name of the rule to which this condition applies. The ``cond`` argument is a string that will be evaluated internally. You put any kind of Boolean argument you want here, or even create a user definable function that returns a Boolean argument (eg, ``'myfunction(arguments)'``). When this condition evaluates as true the string in ``action`` will be evaluated. The string ``SI_TT_BOLUS`` is a prototype function used to modify dosing at the titration points. See the help (``?system_set_tt_cond``) for details on prototype functions and the variables available in the titration environment. This action will define the dosing into the specified state (as defined by ```` in the system file) with the values and times shown. The units here are those specified in the system file and the times are relative to the titration time point. ```{r results="hide", message=FALSE} cfg=system_set_tt_cond(cfg, name = "ivdose", cond = "TRUE", action = "SI_TT_BOLUS[state='Cc', values=500, times=0]", value = "1") ``` Next we simulate the system using the titration function: ```{r results="hide", message=FALSE} som_tt = run_simulation_titrate(parameters, cfg) ``` ```{r results="hide", warning=FALSE, echo=FALSE} # When eval is set to TRUE we save the presimulated results presim$bolus$som_tt$simout = data.frame(ts.days = som_tt$simout$ts.days, Cc = som_tt$simout$Cc) presim$bolus$som_fix$simout = data.frame(ts.days = som_fix$simout$ts.days, Cc = som_fix$simout$Cc) ``` ```{r results="hide", warning=FALSE, echo=FALSE, eval=TRUE} if(presim_loaded){ som_tt = presim$bolus$som_tt som_fix = presim$bolus$som_fix } ``` Now we can overlay the predictions using both methods to show that they produce the same result: ```{r warning=FALSE, message=FALSE, fig.width=7, fig.height=3, eval=TRUE} myfig = ggplot() + geom_line(data=som_fix$simout, aes(x=ts.days, y=Cc, color="Fixed Dosing"), linetype=1) + geom_line(data=som_tt$simout, aes(x=ts.days, y=Cc, color="Titration" ), linetype=2) + scale_colour_manual(values=c("Fixed Dosing"="darkblue", "Titration"="firebrick3")) + theme(legend.title = element_blank()) + theme(legend.position = 'bottom') myfig = prepare_figure('present', myfig) print(myfig) ``` The output of `run_simulation_titrate()` is similar to that of `run_simulation_ubiquity()` with two extra elements in the list that is returned: * ``titration`` - A data frame with a row for each time course output and the following columns for titration rule ``tname``: + ``tt.tname.value`` - Value of the rule for the active condition or -1 if not triggered + ``tt.tname.simtime`` - Simulation time where the last condition became active + ``tt.tname.timescale`` - Simulation time in the time scale the rule was specified in Each of these fields is * ``titration_history`` data frame which contains a summary list of all of the titration events that were triggered. + ``tname`` - Titration rule name + ``value`` - Value of the rule for the active condition + ``simtime`` - Simulation time where the last condition became active + ``timescale`` - Simulation time in the time scale the rule was specified in ## Repeated Infusions (``analysis_repeat_infusion.r``) In this example we want to give a 30 minute infusion at 15 mg/min every two weeks. Similar to the previous example we first do this using the fixed dosing methods: ```{r warning=FALSE, message=FALSE, echo=FALSE} # resetting everything for the repeat infusion example cfg = system_select_set(cfg, "default") parameters = system_fetch_parameters(cfg) cfg=system_set_option(cfg,group = "simulation", option = "solver", value = "lsoda") cfg=system_set_option(cfg, group = "simulation", option = "output_times", seq(0,10*7,.5)) ``` ```{r warning=FALSE, message=FALSE} cfg = system_zero_inputs(cfg) cfg = system_set_rate(cfg, rate = "Dinf", times = c( 0, 30, 20160, 20190, 40320, 40350, 60480, 60510, 80640, 80670), levels = c(15 , 0, 15, 0, 15, 0, 15, 0, 15, 0)) som_fix = run_simulation_ubiquity(parameters, cfg) ``` Now to use the rule-based dosing we enable titration, create the rule (``"ivdose"``) to hold the dosing times, and create the conditions resulting in the dosing. This is all similar to the previous example except we use the ``SI_TT_RATE`` prototype function. The rate and the units of times and levels are the same as those specified in the system file (````). The times here are also relative to the titration time. ```{r warning=FALSE, message=FALSE} cfg=system_set_option(cfg, group = "titration", option = "titrate", value = TRUE) cfg=system_new_tt_rule(cfg, name = "ivdose", times = c(0, 2, 4, 6, 8), timescale = "weeks") ``` ```{r warning=FALSE, message=FALSE} cfg=system_set_tt_cond(cfg, name = "ivdose", cond = "TRUE", action = "SI_TT_RATE[rate='Dinf', times=c(0,30), levels=c(15,0)]", value = "1") ``` ```{r warning=FALSE, message=FALSE} som_tt = run_simulation_titrate(parameters, cfg) ``` ```{r results="hide", warning=FALSE, echo=FALSE} # When eval is set to TRUE we save the presimulated results presim$infusion$som_tt$simout = data.frame(ts.days = som_tt$simout$ts.days, Cc = som_tt$simout$Cc) presim$infusion$som_fix$simout = data.frame(ts.days = som_fix$simout$ts.days, Cc = som_fix$simout$Cc) ``` ```{r results="hide", warning=FALSE, echo=FALSE, eval=TRUE} if(presim_loaded){ som_tt = presim$infusion$som_tt som_fix = presim$infusion$som_fix } ``` ```{r warning=FALSE, message=FALSE, fig.width=7, fig.height=3, eval=TRUE} myfig = ggplot() + geom_line(data=som_fix$simout, aes(x=ts.days, y=Cc, color="Fixed Dosing"), linetype=1) + geom_line(data=som_tt$simout, aes(x=ts.days, y=Cc, color="Titration" ), linetype=2) + scale_colour_manual(values=c("Fixed Dosing"="darkblue", "Titration"="firebrick3")) + theme(legend.title = element_blank()) + theme(legend.position = 'bottom') myfig = prepare_figure('present', myfig) print(myfig) ``` ## Titrated Dosing (``analysis_visit_dosing_titration.r``) In the following example we want to administer a subcutaneous dose (into ``At``) every two weeks (``interval=14``) for 12 weeks. The dose level will be evaluated every 6 months. If the serum concentration (Cc) is less than 900, we will dose at 700 mg. If Cc it is greater than 900 we dose at 600 mg. The relevant code to implement this are shown below: ```{r warning=FALSE, message=FALSE, echo=FALSE} cfg = system_select_set(cfg, "default") parameters = system_fetch_parameters(cfg) cfg=system_set_option(cfg,group = "simulation", option = "solver", value = "lsoda") cfg=system_set_option(cfg, group = "simulation", option = "output_times", seq(0,28*7*4,8)) cfg=system_set_option(cfg, group = "titration", option = "titrate", value = TRUE) ``` ```{r warning=FALSE, message=FALSE} cfg=system_new_tt_rule(cfg, name = "ivdose", times = c(0, 6, 12, 18, 24), timescale = "months") cfg=system_set_tt_cond(cfg, name = "ivdose", cond = "Cc < 900", action = "SI_TT_BOLUS[state='At', values=700, times=0, repdose='last', number=11, interval=14]", value = "700") cfg=system_set_tt_cond(cfg, name = "ivdose", cond = "Cc > 900", action = "SI_TT_BOLUS[state='At', values=600, times=0, repdose='last', number=11, interval=14]", value = "600") som_tt = run_simulation_titrate(parameters, cfg) ``` ```{r results="hide", warning=FALSE, echo=FALSE} # When eval is set to TRUE we save the presimulated results presim$visit$som_tt$simout = data.frame(ts.days = som_tt$simout$ts.days, ts.weeks = som_tt$simout$ts.weeks, Cc = som_tt$simout$Cc) presim$visit$som_tt$titration = som_tt$titration[1:100,] presim$visit$som_tt$titration_history = som_tt$titration_history ``` ```{r results="hide", warning=FALSE, echo=FALSE, eval=TRUE} if(presim_loaded){ som_tt = presim$visit$som_tt } ``` ```{r warning=FALSE, message=FALSE, fig.width=7, fig.height=3, eval=TRUE} myfig = ggplot() + geom_line(data=som_tt$simout, aes(x=ts.weeks, y=Cc), color="blue") myfig = prepare_figure('present', myfig) print(myfig) ``` Initially the concentration in the serum is 0 (``Cc < 900``), so dosing starts at 700. After six months the titration rule is triggered again. At this point the serum concentration is greater greater than 900 so the dose level is reduced to 600. All subsequent measurements are greater than 900 so the dose remains at 600. *Titration output: som_tt$titration* ```{r echo=FALSE, fig.align="center", eval=TRUE} rhandsontable(som_tt$titration, width=550) ``` *Titration output: som_tt$titration_history* ```{r echo=FALSE, fig.align="center", eval=TRUE} rhandsontable(som_tt$titration_history, width=550) ``` ## Monte Carlo Simulations (``analysis_visit_dosing_titration_stochastic.r``) This example is similar to the last except here we are performing population simulations (20 subjects). The same function (`simulate_subjects()`) is used but setting the ``'titrate'`` option to ``TRUE`` tells that function to use the rule-based dosing. ```{r echo=TRUE, message=FALSE, results=FALSE, warning=FALSE} cfg = system_set_option(cfg, group="stochastic", option="nsub", value=20) som= simulate_subjects(parameters, cfg) ``` ```{r results="hide", warning=FALSE, echo=FALSE} # When eval is set to TRUE we save the presimulated results sdf = som_to_df(cfg, som) sdf = sdf[1:100,] #presim$visit$som = som presim$visit$sdf = sdf ``` ```{r results="hide", warning=FALSE, echo=FALSE, eval=TRUE} if(presim_loaded){ #som = presim$visit$som sdf = presim$visit$sdf } ``` This will add a data a list element called ``som$titration`` with three fields for each titration rule (``tname``): * ``tt.tname.value`` - Value of the rule for the active condition * ``tt.tname.simtime`` - Simulation time where the last condition became active * ``tt.tname.timescale`` - Simulation time in the time scale the rule was specified in Each of these fields is a matrix with an entry for each simulation time (column) and each subject (row). In this example these would be accessed through the following list elements: * *``som$titration$tt.ivdose.simtime``* * *``som$titration$tt.ivdose.value``* * *``som$titration$tt.ivdose.timescaletimescale``* This data structure (``som``) can be collapsed down to a data frame using the ``som_to_df`` function. With titration simulations a column is added for each rule giving the value of that rule at the indicated time (the first 100 rows are shown): ```{r message=FALSE, results=FALSE, eval=FALSE} sdf = som_to_df(cfg, som) ``` ```{r echo=FALSE, eval=TRUE} rhandsontable(sdf, width=600, height=300) ``` ## Performing State/Compartment Resets (``analysis_state_reset.r``) Along with changing dosing at titration time points, it may also be necessary to directly alter state or compartment values. In this last example we will dose every two weeks and on week 3 we will drop the total drug in the system by 1/2. ```{r warning=FALSE, message=FALSE, echo=FALSE} cfg = system_select_set(cfg, "default") parameters = system_fetch_parameters(cfg) cfg=system_set_option(cfg,group = "simulation", option = "solver", value = "lsoda") cfg=system_set_option(cfg, group = "simulation", option = "output_times", seq(0,10*7,1)) cfg=system_set_option(cfg, group = "titration", option = "titrate", value = TRUE) ``` This portion covers the IV dosing component: ```{r message=FALSE, results=FALSE} cfg=system_new_tt_rule(cfg, name = "ivdose", times = c(0, 2, 4, 6, 8), timescale = "weeks") cfg=system_set_tt_cond(cfg, name = "ivdose", cond = 'TRUE', action = "SI_TT_BOLUS[state='Cc', values=500, times=0]", value = "5") ``` And this portion reduces the amount of drug in both ``Cc`` and ``Cp`` by 1/2 at week 3: ```{r message=FALSE, results=FALSE} cfg=system_new_tt_rule(cfg, name = "state_reset", times = c(3), timescale = "weeks") cfg=system_set_tt_cond(cfg, name = "state_reset", cond = 'TRUE', action = "SI_TT_STATE[Cc][0.5*Cc]", value = "0") cfg=system_set_tt_cond(cfg, name = "state_reset", cond = 'TRUE', action = "SI_TT_STATE[Cp][0.5*Cp]", value = "0") som_tt = run_simulation_titrate(parameters, cfg) ``` ```{r results="hide", warning=FALSE, echo=FALSE} # When eval is set to TRUE we save the presimulated results presim$state$som_tt$simout = data.frame(ts.weeks = som_tt$simout$ts.weeks, Cc = som_tt$simout$Cc) ``` ```{r results="hide", warning=FALSE, echo=FALSE, eval=TRUE} if(presim_loaded){ som_tt = presim$state$som_tt } ``` ```{r warning=FALSE, message=FALSE, fig.width=7, fig.height=3, eval=TRUE} myfig = ggplot() + geom_line(data=som_tt$simout, aes(x=ts.weeks, y=Cc), color="red") myfig = prepare_figure('present', myfig) print(myfig) ``` ```{r warning=FALSE, message=FALSE, echo=FALSE} save(presim, file="Titration_presim.RData") ``` ## Contents of ``system.txt`` ```{r echo=FALSE, comment='', message=TRUE, eval=TRUE} cat(readLines(system.file("ubinc", "systems","system-mab_pk.txt", package="ubiquity")), sep="\n") ```