Chapter 9 Identification

9.1 Illustration of the backdoor criterion

Perhaps the most common approach to identifying causal effects in observational research is to condition on possible confounders. The “backdoor” criterion for identifying an effect of \(X\) on \(Y\) involves finding a set of nodes to condition on that collectively block all “backdoor paths” between \(X\) and \(Y\). The intuition is that if these paths are blocked, then any systematic correlation between \(X\) and \(Y\) reflects the effect of \(X\) on \(Y\).

To illustrate the backdoor criterion we want to show that estimates of the effect of \(X\) on \(Y\) are identified if we have data on a node that blocks a backdoor path—\(C\)—but not otherwise. With CausalQueries models however, rather than conditioning on \(C\) we simply include data on \(C\) in our model and update as usual.

model <- make_model("C -> X -> Y <- C")  |>
         set_restrictions("(Y[C=1]<Y[C=0])")


# Four types of data: Large, small, door open, door closed
N <- 10000
df_closed_door_large <- make_data(model, n = N)
df_open_door_large   <- mutate(df_closed_door_large, C = NA)
df_closed_door_small <- df_closed_door_large[sample(N, 200), ]
df_open_door_small   <- df_open_door_large[sample(N, 200), ]

We see that with small \(n\) (200 units), closing the backdoor (by including data on \(C\)) produces a tighter distribution on the ATE. With large \(N\) (10,000 units) the distribution around the estimand collapses when the backdoor is closed but not when it is open.

9.2 Identification: Instruments

We illustrate how you can learn about whether \(X=1\) caused \(Y=1\) by taking advantage of an “instrument,” \(Z\).

We start with a model that builds in the instrumental variables exclusion restriction (no unobserved confounding between \(Z\) and \(Y\), no paths between \(Z\) and \(Y\) except through \(X\)) but does not include a monotonicity restriction (no negative effect of \(Z\) on \(X\)).

model <- make_model("Z -> X -> Y; X <-> Y")  

plot(model)

result <- query_model(
    updated, 
    queries = list(ATE = "c(Y[X=1] - Y[X=0])"), 
    given = list(TRUE, "X[Z=1] > X[Z=0]",  "X==0",  "X==1"),
    using = "posteriors")
model query given using case_level mean sd cred.low.2.5% cred.high.97.5%
model_1 ATE - posteriors FALSE 0.4788 0.1056 0.2612 0.6311
model_1 ATE X[Z=1] > X[Z=0] posteriors FALSE 0.5927 0.0492 0.4943 0.6866
model_1 ATE X==0 posteriors FALSE 0.4701 0.1356 0.1901 0.6557
model_1 ATE X==1 posteriors FALSE 0.4877 0.1281 0.2201 0.6638

We calculate the average causal effect (a) for all (b) for the compliers and (c) conditional on values of \(M\).

We see here that the effects are strongest for the “compliers”—units for whom \(X\) responds positively to \(Z\); in addition they are stronger for the treated than for the untreated. Moreover we see that the posterior variance on the complier average effect is low. If our model also imposed a monotonicity assumption then it would be lower still.

model <- make_model("Z -> X -> Y; X <-> Y")  |>
         set_restrictions(decreasing("Z", "X")) 
model query given using case_level mean sd cred.low.2.5% cred.high.97.5%
model_1 ATE - posteriors FALSE 0.4233 0.1047 0.2236 0.6196
model_1 ATE X[Z=1] > X[Z=0] posteriors FALSE 0.6103 0.0291 0.5539 0.6682
model_1 ATE X==0 posteriors FALSE 0.4169 0.1497 0.1556 0.6737
model_1 ATE X==1 posteriors FALSE 0.4300 0.1451 0.1777 0.6779

9.3 Identification through the frontdoor

A less well known approach to identification uses information on the causal path from \(X\) to \(Y\). Consider the following model:

frontdoor <- make_model("X -> M -> Y; X <-> Y") 

plot(frontdoor)

Although in both the instrumental variables (IV) setup and the frontdoor setup we are trying to deal with confounding between \(X\) and \(Y\), the two differ in that in the IV set up we make use of a variable that is prior to \(X\) whereas in the frontdoor model we make use of a variable between \(X\) and \(Y\). In both cases we need other exclusion restrictions: here we see that there is no unobserved confounding between \(X\) and \(M\) or between \(M\) and \(Y\). Importantly too there is no direct path from \(X\) to \(Y\), only the path that runs through \(M\).

Below we plot posterior distributions given observations on 2000 units, with and without data on \(M\):

The spike on the right confirms that we have identification.

9.4 Simple sample selection bias

Say we are interested in assessing the share of Republicans in a population but Republicans are (possible) systematically likely to be absent from our sample. What inferences can we make given our sample?

We will assume that we know when we have missing data, though of course we do not know the value of the missing data.

To tackle the problem we will include sample selection into our model:

model <- make_model("R -> S") |>
  set_parameters(node = "R", parameters = c(2/3,1/3)) |>
  set_parameters(node = "S", parameters = c(1/3, 0, 1/3, 1/3))

data <- make_data(model, n = 1000) |>
        mutate(R = ifelse(S==0, NA, R ))

From this data and model, the priors and posteriors for population and sample quantities are:

model query given using case_level mean sd cred.low.2.5% cred.high.97.5%
model_1 R==1 - parameters FALSE 0.3333 0.3333 0.3333
model_1 R==1 - priors FALSE 0.4975 0.2908 0.0236 0.9741
model_1 R==1 - posteriors FALSE 0.5093 0.1465 0.2635 0.7523
model_1 R==1 S==1 parameters FALSE 0.5000 0.5000 0.5000
model_1 R==1 S==1 priors FALSE 0.4967 0.3071 0.0186 0.9820
model_1 R==1 S==1 posteriors FALSE 0.5097 0.0245 0.4610 0.5577

For the population average effect we tightened our posteriors relative to the priors, though credibility intervals remain wide, even with large data, reflecting our uncertainty about the nature of selection. Our posteriors on the sample mean are accurate and tight.

Importantly we would not do so well if our data did not indicate that we had missingness.

model query given using case_level mean sd cred.low.2.5% cred.high.97.5%
model_1 R==1 - parameters FALSE 0.3333 0.3333 0.3333
model_1 R==1 - posteriors FALSE 0.5056 0.0076 0.4912 0.5205
model_1 R==1 S==1 parameters FALSE 0.5000 0.5000 0.5000
model_1 R==1 S==1 posteriors FALSE 0.5056 0.0076 0.4912 0.5206

We naively conclude that all cases are sampled and that population effects are the same as sample effects. The problem here arises because the causal model does not encompass the data gathering process.

9.5 Addressing both sample selection bias and confounding

Consider the following model from Bareinboim and Pearl (2016) (their Figure 4C). The key feature is that data is only seen for units with \(S=1\) (\(S\) for sampling).

In this model the relationship between \(X\) and \(Y\) is confounded. Two strategies work to address confounding: controlling for either \(Z\) or for \(W1\) and \(W2\) works. But only the first strategy addresses the sample selection problem properly. The reason is that \(Z\) is independent of \(S\) and so variation in \(Z\) is not affected by selection on \(S\).

selection <- make_model("X <- W1 -> W2 -> X -> Y <- Z -> W2; W1 -> S")

To keep the parameter and type space small we also impose a set of restrictions: \(S\) is non decreasing in \(W_1\), \(X\) is not decreasing in either \(W1\) or \(W2\), \(Y\) is not decreasing \(Z\) or \(X\) and \(X\) affects \(Y\) only if \(Z=1\). \(W_2=1\) if and only if both \(W_1=1\) and \(Z=1\). These all reduce the problem to one with 18 nodal types and 288 causal types.

Worth noting that in this model although selection is related to patterns of confounding, it is not related to causal effects: the effect of \(X\) on \(Y\) is not different from units that are or are not selected.

Given these priors we will assume a true (unknown) data generating process with no effect of \(X\) on \(Y\), in which \(W_1\) arises with a \(1/3\) probability but has a strong positive effect on selection into the sample when it does arise.

The estimand values given the true parameters and priors for this model are as shown below.

Table 9.1: Estimand values
model query given using case_level mean sd cred.low.2.5% cred.high.97.5%
model_1 Y[X=1] - Y[X=0] - parameters FALSE 0.000 0.0000 0.0000
model_1 Y[X=1] - Y[X=0] - priors FALSE 0.333 0.1514 0.0811 0.6563

This confirms a zero true effect, though priors are dispersed, centered on a positive effect.

We can see the inference challenge from observational data using regression analysis with and without conditioning on \(Z\) and \(W_1, W_2\).

Dependent variable:
Y
(1) (2) (3)
X 0.083*** 0.063*** -0.0005
(0.008) (0.015) (0.008)
X:W1_norm 0.149***
(0.030)
X:W2_norm 0.142***
(0.032)
X:Z_norm -0.005
(0.016)
Observations 15,031 15,031 15,031
R2 0.007 0.017 0.104
Adjusted R2 0.007 0.017 0.104
Residual Std. Error 0.498 (df = 15029) 0.496 (df = 15025) 0.473 (df = 15027)
F Statistic 103.800*** (df = 1; 15029) 52.770*** (df = 5; 15025) 582.300*** (df = 3; 15027)
Note: p<0.1; p<0.05; p<0.01

Naive analysis is far off; but even after conditioning on \(W_1, W_2\) we still wrongly infer a positive effect.

Bayesian inferences given different data strategies are shown below:

data mean sd
X,Y 0.0726 0.0143
X,Y, W1, W2 0.0156 0.0065
X, Y, Z 0.0120 0.0051

We see the best performance is achieved for the model with data on \(Z\)—in this case the mean posterior estimate is closest to the truth–0–and the standard deviation is lowest also. However the gains in choosing \(Z\) over \(W1, W2\) are not as striking as in the regression estimates since knowledge of the model structure protects us from error.

9.6 Learning from a collider!

Conditioning on a collider can be a bad idea as it can introduce a correlation between variables that might not have existed otherwise (Elwert and Winship 2014). But that doesn’t mean colliders should be ignored in analysis altogether. For a Bayesian, knowledge of the value of a collider can still be informative.

Pearl describes a model similar to the following as a case for which controlling for covariate \(W\) induces bias in the estimation of the effect of \(X\) on \(Y\), which could otherwise be estimated without bias using simple differences in means.

model <- make_model("X -> Y <- U1 -> W <- U2 -> X") |>
  
  set_restrictions(labels = list(Y = c("0001", "1111"), W = "0001"), keep = TRUE) |>
  set_restrictions("(X[U2=1]<X[U2=0])") |>
  set_parameters(node = "U1",  parameters = c(1/4, 3/4)) |>
  set_parameters(node = "Y", parameters = c(2/3, 1/3))
## Warning in make_par_values(model, alter = "param_value", x = parameters, : A specified condition matches multiple parameters. In these cases it is unclear which parameter value should be assigned to which parameter. Assignment thus defaults to the order in which parameters appear in 'parameters_df'.
##               
##  We advise checking that parameter assignment was carried out as you intended.

## Warning in make_par_values(model, alter = "param_value", x = parameters, : A specified condition matches multiple parameters. In these cases it is unclear which parameter value should be assigned to which parameter. Assignment thus defaults to the order in which parameters appear in 'parameters_df'.
##               
##  We advise checking that parameter assignment was carried out as you intended.
plot(model)

data <- make_data(model, 
                  n = 25000, 
                  vars = c("W", "X", "Y"), 
                  using = "parameters")

The effect of \(X\) on \(Y\) is .5 but average effects as well as the probability of causation, are different for units with \(W=0\) and \(W=1\) (this, even though \(W\) does not affect \(Y\)):

model query given using case_level mean sd cred.low.2.5% cred.high.97.5%
model_1 Y(1)-Y(0) - parameters FALSE 0.5000 0.5000 0.5000
model_1 Y(1)-Y(0) W==0 parameters FALSE 0.4000 0.4000 0.4000
model_1 Y(1)-Y(0) W==1 parameters FALSE 0.6667 0.6667 0.6667
model_1 Y(1)-Y(0) X==1 & Y==1 parameters FALSE 0.6000 0.6000 0.6000
model_1 Y(1)-Y(0) X==1 & Y==1 & W==0 parameters FALSE 0.5000 0.5000 0.5000
model_1 Y(1)-Y(0) X==1 & Y==1 & W==1 parameters FALSE 0.6667 0.6667 0.6667

These are the quantities we seek to recover. The ATE can be gotten fairly precisely in a simple regression. But controlling for \(W\) introduces bias in the estimation of this effect (whether done using a simple control or an interactive model):

Dependent variable:
Y
(1) (2) (3)
X 0.498*** 0.453*** 0.458***
(0.005) (0.005) (0.005)
W 0.187***
(0.006)
W_norm -0.008
(0.008)
X:W_norm 0.346***
(0.011)
Constant 0.334*** 0.285*** 0.333***
(0.004) (0.004) (0.004)
Observations 25,000 25,000 25,000
R2 0.255 0.286 0.313
Adjusted R2 0.255 0.286 0.313
Residual Std. Error 0.426 (df = 24998) 0.417 (df = 24997) 0.409 (df = 24996)
F Statistic 8,539.000*** (df = 1; 24998) 5,013.000*** (df = 2; 24997) 3,795.000*** (df = 3; 24996)
Note: p<0.1; p<0.05; p<0.01

How does the Bayesian model do, with and without data on \(W\)?

Without \(W\) we have:

Table 9.2: Collider excluded from model
model query given using case_level mean sd cred.low.2.5% cred.high.97.5%
model_1 Y(1)-Y(0) - posteriors FALSE 0.5078 0.0054 0.4972 0.5188
model_1 Y(1)-Y(0) W==0 posteriors FALSE 0.3727 0.1200 0.0473 0.5058
model_1 Y(1)-Y(0) W==1 posteriors FALSE 0.6743 0.0043 0.6661 0.6828
model_1 Y(1)-Y(0) X==1 & Y==1 posteriors FALSE 0.6093 0.0054 0.5990 0.6198
model_1 Y(1)-Y(0) X==1 & Y==1 & W==0 posteriors FALSE 0.4314 0.1723 0.0246 0.6090
model_1 Y(1)-Y(0) X==1 & Y==1 & W==1 posteriors FALSE 0.6743 0.0043 0.6661 0.6828

Thus we estimate the treatment effect well. What’s more we can estimate the probability of causation when \(W=1\) accurately, even though we have not observed \(W\). The reason is that if \(W=1\) then, given the model restrictions, we know that both \(U_1=1\) and \(U_2=1\) which is enough. We are not sure however what to infer when \(W=0\) since this could be due to either \(U_1=0\) or \(U_2=0\).

When we incorporate data on \(W\) our posteriors are:

Table 9.3: Collider included in model
model query given using case_level mean sd cred.low.2.5% cred.high.97.5%
model_1 Y(1)-Y(0) - posteriors FALSE 0.5075 0.0054 0.4969 0.5181
model_1 Y(1)-Y(0) W==0 posteriors FALSE 0.4067 0.0066 0.3940 0.4198
model_1 Y(1)-Y(0) W==1 posteriors FALSE 0.6741 0.0042 0.6661 0.6823
model_1 Y(1)-Y(0) X==1 & Y==1 posteriors FALSE 0.6090 0.0053 0.5986 0.6193
model_1 Y(1)-Y(0) X==1 & Y==1 & W==0 posteriors FALSE 0.5066 0.0078 0.4912 0.5216
model_1 Y(1)-Y(0) X==1 & Y==1 & W==1 posteriors FALSE 0.6741 0.0042 0.6661 0.6823

We see including the collider does not induce error in estimation of the ATE, even though it does in a regression framework. Where we do well before we continue to do well. However the new information lets us improve our model and, in particular, we see that we now get a good and tight estimate for the probability that \(X=1\) caused \(Y=1\) in a case where \(W=0\).

In short, though conditioning on a collider induces error in a regression framework; including the collider as data for updating our causal model doesn’t hurt us and can help us.

References

Bareinboim, Elias, and Judea Pearl. 2016. “Causal Inference and the Data-Fusion Problem.” Proceedings of the National Academy of Sciences 113 (27): 7345–52.
Elwert, Felix, and Christopher Winship. 2014. “Endogenous Selection Bias: The Problem of Conditioning on a Collider Variable.” Annual Review of Sociology 40: 31–53.