6.7. Project: Gompertz Model for Tumor Growth#
4/19/2022. Released to students
4/20/2022. Corrected spelling of radiation. Added preamble for Google Colab
4/20/2022. Corrected text for alpha and beta. alpha = 0.08, beta = 0.008
4/21/2022. Added comments on grading for problem 3. Deleted unused code cells at the end of the notebook.
For this project you will apply the modeling and dynamic optimization skills we have been learning in class to devise therapeutic strategies for treating tumors.
The Gompertz model was developed in the 19th century as actuarial model for human mortality, then adapted in the 1960’s as a successful model to describe the growth of tumors. Conceptually, the model describes the growth cycle of a population of cancer cells. In practice the model is normally calibrated to represent the measured size of a tumor. Let \(N\) be the diameter of the tumor, then the usual form of the Gompertz model is given by
where \(\rho\) is a growth constant related to the proliferative ability of the cells, and \(K\) is carrying capacity defined as the maximum size that can be reached with available nutrients. For this project, we will use a slightly modified model that generally behaves better for optimization studies
This is the generalized logistic equation. The parameters have the same interpretation.
Geng, C., Paganetti, H., & Grassberger, C. (2017). Prediction of treatment response for combined chemo-and radiation therapy for non-small cell lung cancer patients using a bio-mathematical model. Scientific reports, 7(1), 1-12. https://www.ncbi.nlm.nih.gov/pmc/articles/PMC5648928/
6.7.1. Problem 1. Simulation (10 pts).#
Staging is a step in the diagnostic process that accounts for a complex array of biomedical features to determine the current state of disease and to guide subsequent therapeutic decisions. For this project we will use tumor size as a measure of disease stage. The first problem examines the progress of the disease in the absence of any therapy.
Given an initial tumor size at the detectable limit 0.3 cm, solve for the time for the disease to evolve to Stage 1, Stage 2, etc., as outlined in the table below. You may use any solution technique of your choice, such as root find or adding decision variables to the Pyomo model. Use parameter values \(\rho = 0.02\) and \(K = 20.0\)
Size |
Stage |
Day |
---|---|---|
< 3 mm |
Undetectable |
0 |
1.2 cm |
Stage 1 |
? |
3.5 cm |
Stage 2 |
? |
6.9 cm |
Stage 3 |
? |
9.7 cm |
Stage 4 |
? |
13.0 cm |
Death |
? |
Your solution to this problem will be graded on
(3 pts) presenting a viable solution strategy,
(5 pts) execution of the strategy in code,
(2 pts) the numerical solution.
6.7.1.1. Your Solution#
Place your comments here.
Use the following Pyomo model as a starting point for your solution.
import pyomo.environ as pyo
import pyomo.dae as dae
import matplotlib.pyplot as plt
import numpy as np
rho = 0.02
K = 20.0
def growth_model(IC=1.0, tf=400):
m = pyo.ConcreteModel()
m.t = dae.ContinuousSet(bounds=(0, tf))
m.N = pyo.Var(m.t, domain=pyo.NonNegativeReals)
m.dNdt = dae.DerivativeVar(m.N)
m.N[0].fix(IC)
@m.Constraint(m.t)
def gompertz(m, t):
return m.dNdt[t] == rho * m.N[t] * (1 - m.N[t]/K)
pyo.TransformationFactory('dae.finite_difference').apply_to(m, nfe=2*int(tf), scheme='FORWARD')
pyo.SolverFactory('ipopt').solve(m)
return m
m = growth_model(0.3, 400)
fig, *ax = plt.subplots(figsize=(12, 5))
ax[0].plot(m.t, [m.N[t]() for t in m.t], label='untreated growth')
ax[0].set_xlabel('Days')
ax[0].set_ylabel('Size (cm)')
ax[0].set_title('Gompertz model')
ax[0].grid(True)
6.7.2. Problem 2. Radiation Therapy (10pts)#
Radiation therapy is a widely use therapy intervention for many solid tumors. A model for the effect of radiation is given by
where the second term modifies the Gompertz growth model to include radiation effects. (\(\alpha = 0.08\), \(\beta = 0.008\))
Variable \(d(t)\) is dosage measured in Gray (Gy), the standard SI units for absorbed dose of ionizing radiation. The radiation therapy is limited to a daily limit of 0.8 Gray/day and a total limit of 60 Gray. As a head start, the following Pyomo model is given which incorporates the constraint
The main challenge of this problem is to identify and implement a meaningful objective function.
Modify the Pyomo model below to include the \(d \leq 0.8\) Gy constraint on daily dose.
Consider a therapy horizon of 400 days. In either words or with a mathematical expression, write down an objective function in the cell below.
Implement and test your objective function for a patient diagnosed with Stage 2 disease. You may need to modify and test your objective until you arrive at a relevant result. There is no “right” answer, but some answers are better than others. Will radiation therapy alone provide adequate treatment subject the constraints and given model parameters?
Your solution will be graded as follows:
Written rationale for the choice of objective. (4 pts.)
Correctly implemented constraints on radiation. (2 pts.)
Code and implementation. (4 pts.)
import pyomo.environ as pyo
import pyomo.dae as dae
import matplotlib.pyplot as plt
import numpy as np
alpha = 0.06
beta = 0.006
def radiation_model(IC=1.0, tf=400):
m = pyo.ConcreteModel()
m.t = dae.ContinuousSet(bounds=(0, tf))
m.N = pyo.Var(m.t, domain=pyo.NonNegativeReals)
m.d = pyo.Var(m.t, domain=pyo.NonNegativeReals)
m.dNdt = dae.DerivativeVar(m.N)
m.N[0].fix(IC)
@m.Integral(m.t)
def doseage(m, t):
return m.d[t]
@m.Constraint()
def total_dose(m):
return m.doseage <= 60
@m.Constraint(m.t)
def gompertz(m, t):
return m.dNdt[t] == rho * m.N[t] * (1 - m.N[t]/K) - (alpha * m.d[t] + beta * m.d[t]**2) * m.N[t]
pyo.TransformationFactory('dae.finite_difference').apply_to(m, nfe=1*int(tf), scheme='FORWARD')
pyo.SolverFactory('ipopt').solve(m)
return m
m = radiation_model(3.5, 400)
fig, ax = plt.subplots(2, 1, figsize=(12, 5), sharex=True)
ax[0].plot(m.t, [m.N[t]() for t in m.t], label='untreated growth')
ax[0].set_ylabel('Size (cm)')
ax[0].set_title('Gompertz model')
ax[0].grid(True)
ax[1].plot(m.t, [m.d[t]() for t in m.t], label='radiation')
ax[1].set_ylabel('dose [Gr/day]')
ax[1].set_xlabel('Days')
ax[1].grid(True)
print(m.doseage())
59.59860467993624
6.7.3. Problem 3. Combined Radiation and Chemo Therapy (20 pts)#
Another common treatment is a combination of chemotherapy and radiation. The chemotherapy is incorporated in the model as
where the parameter \(\gamma = 0.01\). The chemotherapeutic agent is toxic and therefore limited to 4 units per day, and a total 100 units over 100 days. The drug as a half-life of 1 day.
Using the Pyomo model below as a starting point, add additional decision variables
m.u[t]
to model the administration of the chemotherapeutic agent, andm.C[t]
to model the blood concentration of the agent. You will need to add an additional differential equation to model the half-life of the agent in the bloodstream.Add constraints all necessary constraints to restrict the total radiation and chemotherapeutic dosages.
Add a suitable objective functions.
Test the performance of your approach to a patient presenting with Stage 3 disease.
Grading:
The mathematical model developed for \(C(t)\) and incorporated into Pyomo. Be careful to get the half-life right. (5 pts)
Establishing constraints for radiation and chemotherapy (5 pts)
Discussion and development of an objective function (5 pts)
Pyomo modeling and simulation of results. (5 pts)
import pyomo.environ as pyo
import pyomo.dae as dae
import matplotlib.pyplot as plt
import numpy as np
alpha = 0.06
beta = 0.006
def radiation_model(IC=1.0, tf=400):
m = pyo.ConcreteModel()
m.t = dae.ContinuousSet(bounds=(0, tf))
m.N = pyo.Var(m.t, domain=pyo.NonNegativeReals)
m.d = pyo.Var(m.t, domain=pyo.NonNegativeReals)
m.dNdt = dae.DerivativeVar(m.N)
m.N[0].fix(IC)
@m.Integral(m.t)
def doseage(m, t):
return m.d[t]
@m.Constraint()
def total_dose(m):
return m.doseage <= 60
@m.Constraint(m.t)
def gompertz(m, t):
return m.dNdt[t] == rho * m.N[t] * (1 - m.N[t]/K) - (alpha * m.d[t] + beta * m.d[t]**2) * m.N[t]
pyo.TransformationFactory('dae.finite_difference').apply_to(m, nfe=1*int(tf), scheme='FORWARD')
pyo.SolverFactory('ipopt').solve(m)
return m
m = radiation_model(3.5, 400)
fig, ax = plt.subplots(3, 1, figsize=(12, 5), sharex=True)
ax[0].plot(m.t, [m.N[t]() for t in m.t], label='untreated growth')
ax[0].set_ylabel('Size (cm)')
ax[0].set_title('Gompertz model')
ax[0].grid(True)
ax[1].plot(m.t, [m.d[t]() for t in m.t], label='radiation')
ax[1].set_ylabel('dose [Gr/day]')
ax[1].set_xlabel('Days')
ax[1].grid(True)
#ax[2].plot(m.t, [m.u[t]() for t in m.t], label='chemotherapy')
ax[2].set_ylabel('dose [Gr/day]')
ax[2].set_xlabel('Days')
ax[2].grid(True)
print(m.doseage())
59.59860467993624