library(Colossus)
library(data.table)
library(survival)
library(dplyr)
#> 
#> Attaching package: 'dplyr'
#> The following objects are masked from 'package:data.table':
#> 
#>     between, first, last
#> The following objects are masked from 'package:stats':
#> 
#>     filter, lag
#> The following objects are masked from 'package:base':
#> 
#>     intersect, setdiff, setequal, unionOne alternative regression method in Colossus is matched case-control logistic regression. The theory is presented in the following section.
Suppose we have matched case-control data and divide our data into each matched set. Each set has \(m\) cases and \(n\) records. We denote the relative risk for individual \(i\) in the set by \(r_i\). We can calculate the probability of case exposures conditional on all exposures in the set by taking the ratio of the product of relative risks in the cases to the sum of the product of relative risks for every way of selecting \(m\) individuals from the \(n\) at risk.
\[ \begin{aligned} \frac{\prod_{i}^{m} r_i}{\sum_{c \in R} \left ( \prod_{j=1}^{m} r_{c_j} \right )} \\ L = \sum_{i=1}^{m} log(r_i) - log \left ( \sum_{c \in R} \left ( \prod_{j=1}^{m} r_{c_j} \right ) \right ) \end{aligned} \]
Using the methods presented in Gail et al. (1981) we can calculate the combination of all \(n!/m!(n-m)!\) ways to select \(m\) items with a more manageable recursive formula \(B(m,n)\).
\[ \begin{aligned} B(m, n) = \sum_{c \in R} \left ( \prod_{j=1}^{m} r_{c_j} \right ) \\ B(m,n) = B(m, n-1) + r_n B(m-1, n-1) \\ B(m,n) = \begin{cases} \sum_{j}^{n} r_j & m = 1 \\ 0 & m > n \end{cases} \end{aligned} \]
We can then directly solve for the first and second derivatives and their recursive formula.
\[ \begin{aligned} \frac{\partial r_i}{\partial \beta_\mu} =: r_{i}^{\mu} \\ \frac{\partial B(m,n)}{\partial \beta_\mu} = B^{\mu}(m, n) = \sum_{c \in R} \left [ \left ( \sum_{j=1}^{m} \frac{r_{c_j}^{\mu}}{r_{c_j}} \right ) \prod_{j=1}^{m} r_{c_j} \right ] \\ B^{\mu}(m,n) = B^{\mu}(m, n-1) + r_n B^{\mu}(m-1, n-1) + r_n^{\mu} B(m-1, n-1) \\ B^{\mu}(m,n) = \begin{cases} \sum_{j}^{n} r_j^{\mu} & m = 1 \\ 0 & m > n \end{cases} \end{aligned} \]
\[ \begin{aligned} \frac{\partial^2 r_i}{\partial \beta_\mu \partial \beta_\nu} =: r_{i}^{\mu,\nu} \\ \frac{\partial^2 B(m,n)}{\partial \beta_\mu \partial \beta_\nu} = B^{\mu,\nu}(m, n) = \sum_{c \in R} \left [ \left ( \sum_{j=1}^{m} \frac{r_{c_j}^{\mu,\nu}}{r_{c_j}} + \left ( \sum_{j=1}^{m} \frac{r_{c_j}^{\mu}}{r_{c_j}} \right ) \left ( \sum_{j=1}^{m} \frac{r_{c_j}^{\nu}}{r_{c_j}} \right ) - \sum_{j=1}^{m} \frac{r_{c_j}^{\mu}}{r_{c_j}} \frac{r_{c_j}^{\nu}}{r_{c_j}} \right ) \prod_{j=1}^{m} r_{c_j} \right ] \\ B^{\mu,\nu}(m,n) = B^{\mu,\nu}(m, n-1) + r_n^{\mu,\nu} B(m-1, n-1) + r_n^{\nu} B^{\mu}(m-1, n-1) + r_n^{\mu} B^{\nu}(m-1, n-1) + r_n B^{\mu,\nu}(m-1, n-1) \\ B^{\mu,\nu}(m,n) = \begin{cases} \sum_{j}^{n} r_j^{\mu,\nu} & m = 1 \\ 0 & m > n \end{cases} \end{aligned} \]
Finally, these expressions for \(B(m,n)\) can be substituted into the equations for the contribution of Log-Likelihood and its derivatives from each matched set. The model is then optimized via the same methods as the other regression models.
\[ \begin{aligned} L_{set} = \sum_{i=1}^{m} log(r_i) - log \left ( B(m,n) \right ) \\ L^{\mu}_{set} = \sum_{i=1}^{m} \frac{r_i^{\mu}}{r_i} - \frac{B^{\mu}(m,n)}{B(m,n)} \\ L^{\mu, \nu}_{set} = \sum_{i=1}^{m} \left ( \frac{r_i^{\mu,\nu}}{r_i} - \frac{r_i^{\mu}}{r_i}\frac{r_i^{nu}}{r_i} \right ) - \left ( \frac{B^{\mu,\nu}(m,n)}{B(m,n)} - \frac{B^{\mu}(m,n)}{B(m,n)}\frac{B^{\nu}(m,n)}{B(m,n)} \right ) \end{aligned} \]
It is important to note that the recursive formula calculation can quickly become time-consuming, particularly if there is a large number of cases. To make the matched case-control method generally applicable, the likelihood function can be changed to a logistic regression model in matched sets with a large number of cases. In general, the matched case-control regression function adds an item to the model control list, “cond_thres”, to set the threshold to switch to a logistic regression model.
The logistic loglikelihood is defined by treating the matched case-control data as single trial data. The likelihood is a function of the event status (\(\theta_i\)) and odds ratio (\(O_i\)). The odds ratio for any row is calculated as the product of the odds ratio for the matched set (\(O_s\)) and the relative risk for the row (\(r_i\)). Behind the scenes, Colossus optimizes both the model parameters (\(\vec{\beta}\)) for the relative risk as well as a logistic model for the matched set odds ratios (\(O_s=e^{\alpha_s}\)).
\[ \begin{aligned} L_i = y_i ln(O_s r_i) - ln(1+ O_s r_i) \\ \frac{\partial L_i}{\partial \beta_j} = y_i \frac{r^j_i}{r_i} - O_s \frac{r^j_i}{1 + O_s r_i}\\ \frac{\partial L_i}{\partial \alpha_s} = (y_i - 1) + \frac{1}{1 + O_s r_i} \\ \frac{\partial^2 L_i}{\partial \beta_j \partial \beta_k} = y_i \left ( \frac{r^{j,k}_i}{r_i} - \frac{r^j_i}{r_i} \frac{r^k_i}{r_i} \right ) - O_s \left ( \frac{r^{j,k}_i}{1 + O_s r_i} - O_s \frac{r^j_i}{1 + O_s r_i} \frac{r^k_i}{1 + O_s r_i} \right ) \\ \frac{\partial^2 L_i}{\partial \beta_j \partial \alpha_s} = \frac{-O_s r^j_i}{1 + O_s r_i} \\ \frac{\partial^2 L_i}{\partial \alpha_s^2} = \frac{-O_s r_i}{1 + O_s r_i} \end{aligned} \]
The following provides a basic example of how to use the matched case-control regression function. The data used is from a lung cancer study, included in the R survival library. Slight adjustments were made to make the data line up with data included with 32-bit Epicure, for comparison. In short, we want to model the effects of treatment and Karnofsky performance score on sets matched by cancer cell type.
#
data(cancer, package = "survival")
df <- veteran
# Make the same adjustments as Epicure example 6.5
karno <- df$karno
karno[93] <- 20
df$karno <- karno
df$trt <- df$trt - 1
df$trt <- as.integer(df$trt == 0)
cell_string <- df$celltype
cell <- case_when(
  cell_string == "squamous" ~ 1,
  cell_string == "smallcell" ~ 2,
  cell_string == "adeno" ~ 3,
  cell_string == "large" ~ 0
)
df$cell <- cell
df$karno50 <- df$karno - 50
# Convert the cell column into factor columns
fcols <- c("cell")
val <- factorize(df, fcols) # Colossus function
df <- val$df
t0 <- "%trunc%"
t1 <- "time"
event <- "status"
names <- c(
  "karno50", "trt"
)
tform_1 <- c(
  "loglin", "loglin"
)
term_n <- c(0, 0)
a_n <- c(0.1, 0.1)In the following examples, we are using matching by strata and changing the conditional threshold. In the first case, every matched set uses the recursive formula. In the second case, one set uses the simplified formula. Finally in the third case, every set uses the simplified formula. In all cases, the regression returns the typical output and can be summarized similarly to other regression function outputs.
control <- list(verbose = 2, maxiters = c(25, 25))
model_control <- list("strata" = T, "conditional_threshold" = 100)
e0 <- RunCaseControlRegression_Omnibus(
  df, t0, t1, event,
  names = names, tform = tform_1,
  strat_col = "cell", model_control = model_control,
  control = control, term_n = term_n, a_n = a_n
)
model_control <- list("strata" = T, "conditional_threshold" = 40)
e1 <- RunCaseControlRegression_Omnibus(
  df, t0, t1, event,
  names = names, tform = tform_1,
  strat_col = "cell", model_control = model_control,
  control = control, term_n = term_n, a_n = a_n
)
model_control <- list("strata" = T, "conditional_threshold" = 0)
e2 <- RunCaseControlRegression_Omnibus(
  df, t0, t1, event,
  names = names, tform = tform_1,
  strat_col = "cell", model_control = model_control,
  control = control, term_n = term_n, a_n = a_n
)
Interpret_Output(e0)
#> |-------------------------------------------------------------------|
#> Final Results
#>    Covariate Subterm Term Number Central Estimate Standard Error 1-tail p-value
#>       <char>  <char>       <int>            <num>          <num>          <num>
#> 1:   karno50  loglin           0      -0.04350168     0.02315817     0.03015916
#> 2:       trt  loglin           0      -0.36615825     0.73703418     0.30966521
#> 
#> Matched Case-Control Model Used
#> Deviance: 49.86
#> 0 out of 4 matched sets used Unconditional Likelihood
#> Iterations run: 18
#> maximum step size: 9.54e-07, maximum first derivative: 3.79e+02
#> Analysis did not converge, check convergence criteria or run further
#> Run finished in 0.25 seconds
#> |-------------------------------------------------------------------|
Interpret_Output(e1)
#> |-------------------------------------------------------------------|
#> Final Results
#>    Covariate Subterm Term Number Central Estimate Standard Error 1-tail p-value
#>       <char>  <char>       <int>            <num>          <num>          <num>
#> 1:   karno50  loglin           0      -0.04386339     0.02048505     0.01612741
#> 2:       trt  loglin           0      -0.37043486     0.57345538     0.25914946
#> 
#> Matched Case-Control Model Used
#> Deviance: 52.75
#> 1 out of 4 matched sets used Unconditional Likelihood
#> Iterations run: 19
#> maximum step size: 9.54e-07, maximum first derivative: 4.35e+02
#> Analysis did not converge, check convergence criteria or run further
#> Run finished in 0.2 seconds
#> |-------------------------------------------------------------------|
Interpret_Output(e2)
#> |-------------------------------------------------------------------|
#> Final Results
#>    Covariate Subterm Term Number Central Estimate Standard Error 1-tail p-value
#>       <char>  <char>       <int>            <num>          <num>          <num>
#> 1:   karno50  loglin           0      -0.04487003     0.01633575    0.003009551
#> 2:       trt  loglin           0      -0.37735968     0.55683957    0.248986940
#> 
#> Matched Case-Control Model Used
#> Deviance: 59.72
#> 4 out of 4 matched sets used Unconditional Likelihood
#> Iterations run: 21
#> maximum step size: 9.54e-07, maximum first derivative: 7.06e+02
#> Analysis did not converge, check convergence criteria or run further
#> Run finished in 0.22 seconds
#> |-------------------------------------------------------------------|