# bayesAB 0.7.0 + A Primer on Priors

**Frank Portman**, and kindly contributed to R-bloggers]. (You can report issue about the content on this page here)

Want to share your content on R-bloggers? click here if you have a blog, or here if you don't.

## bayesAB 0.7.0

Quick announcement that my package for Bayesian AB Testing, bayesAB, has been updated to 0.7.0 on CRAN. Some improvements on the backend as well a few tweaks for a more fluid UX/API. Some links:

Now, on to the good stuff.

## Why should we care about priors?

Most questions I’ve gotten since I released bayesAB have been along the lines of:

- Why/how is Bayesian AB testing better than Frequentist hypothesis AB testing?
- Why do I need priors?
- Do I really really really need priors?
- How do I choose priors?

Question 1 has a few objective and a few subjective answers to it. The main benefits are ones that I’ve already highlighted in the README/vignette of the bayesAB package. To briefly summarize, we get direct probabilities for A > B (rather than p-values) and distributions over the parameter estimates rather than point estimates. Finally, we can also leverage **priors** which help with the low sample size and low base rate problems.

To start, let’s go back to what a **prior** actually is in a Bayesian context. There are countless mathematical resources out there (including part of my previous blog post) so I’ll only about this conceptually. Simply put, a **prior** lets you specify some sort of, ahem, *prior* information about a certain parameter so that the end **posterior** on that parameter encapsualtes both the **data** you saw and the **prior** you inputted. Priors can come from a variety of places including past experiments, literature, and domain expertise into the problem. See this blogpost for a great example of somebody combining their own past data and literature to form very strong priors.

**Priors** can be weak or strong. The weakest prior will be completely **objective** and thus assign an equal probability to each value for the parameter. Examples of this include a Beta(1, 1) prior for the Bernoulli distribution. In these cases, the **posterior** distribution is completely reliant on the **data**. A strong **prior** will convey a very precise belief as to where a parameter’s values may lie. For example:

1 2 3 | library(bayesAB) plotBeta(1000, 1000) |

The stronger the **prior** the more say it has in the **posterior** distribution. Of course, according to the Bernstein–von Mises theorem the **posterior** is effectively independent of the **prior** once a large enough sample size has been reached for the **data**. How quickly this is the case, depends on the strength of your **prior**.

Do you need (weak/strong) **priors**? Not necessarily. You can still leverage the interpretability benefits of Bayesian AB testing even without priors. At worst, you’ll also get slightly more pertinent results since you can parametrize your metrics as the appropriate distribution random variable. However, without **priors** of some kind (and to be clear, not random bullshit priors either) you run into similar issues as with Frequentist AB testing, namely Type 1 and Type 2 errors. A Type 1 error is calling one version better when it really isn’t, and a Type 2 error is calling a better version equal or worse. Both typically arise from low sample size/base rate and are controlled by reaching appropriate sample size as per a power calculation.

### So what can we do?

Have no fear! Even without good and/or strong **priors** there are still ways to control for false positives and all that good stuff. We use something called **Expected Posterior Loss** or “based on the current winner, what is the expected loss you would see should you choose wrongly”. If this value is lower than your **threshold of caring** (`abs(A - b)`

) then you can go ahead and call your test. This value implictly encompasses the uncertainty about your **posteriors**.

Okay cool, that roughly answers Questions 1-4 in some order.

## Simulation

Let’s do a quick simulation to illustrate some of the above points. Let’s make three examples: weak priors, strong priors, and diffuse priors (quick tip: the Jeffrey’s Prior of a Gamma distribution is Gamma(eps, eps) where eps is smallllll). We’ll be taking 2 x 100 samples from a Poisson distribution with the same $\lambda$ parameters. The strong and weak priors will be centered around this value of 2.3.

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 | library(magrittr) n <- 1e3 out_weaker_priors <- rep(NA, n) out_stronger_priors <- rep(NA, n) out_diffuse <- rep(NA, n) getProb <- function(x) summary(x)$probability$Lambda for(i in 1:n) { A <- rpois(100, 2.3) B <- rpois(100, 2.3) out_weaker_priors[i] <- bayesTest(A, B, priors = c('shape' = 23, 'rate' = 10), distribution = 'poisson') %>% getProb out_stronger_priors[i] <- bayesTest(A, B, priors = c('shape' = 230, 'rate' = 100), distribution = 'poisson') %>% getProb out_diffuse[i] <- bayesTest(A, B, priors = c('shape' = 0.00001, 'rate' = 0.00001), distribution = 'poisson') %>% getProb } out_weaker_priors <- ifelse(out_weaker_priors <= 0.05 | out_weaker_priors >= .95, 1, 0) out_stronger_priors <- ifelse(out_stronger_priors <= 0.05 | out_stronger_priors >= .95, 1, 0) out_diffuse <- ifelse(out_diffuse <= 0.05 | out_diffuse >= .95, 1, 0) |

Now, A and B shouldn’t have any difference between the two but occasionally we will see a Type 1 error. That’s what the bottom 3 lines are doing. If P(A > B) is <=0.05 or >= .95 we call one of the recipes “significantly” better. Observe what happens with each case of prior.

1 | mean(out_weaker_priors) |

1 | ## [1] 0.081 |

1 | mean(out_stronger_priors) |

1 | ## [1] 0.02 |

1 | mean(out_diffuse) |

1 | ## [1] 0.095 |

The diffuse priors have the most Type 1 errors, followed by the weak priors, followed by the strong priors; to be expected.

Finally, we can fit another bayesTest (:D) to determine whether the differences between Type 1 error percents across priors are different from one another.

1 2 3 4 | t1 <- bayesTest(out_diffuse, out_weaker_priors, priors = c('alpha' = 1, 'beta' = 1), distribution = 'bernoulli') t2 <- bayesTest(out_diffuse, out_stronger_priors, priors = c('alpha' = 1, 'beta' = 1), distribution = 'bernoulli') plot(t1) |

1 | plot(t2, priors = FALSE) |

As we can see, it’s somewhat clear that the diffuse is worse than the weak and very clear that the diffuse is worse than the stronger priors. Note that in our case I use a diffuse prior of Beta(1, 1) since I have no idea what’s normal going into this simulation.

Finally we can check the output of `summary`

to see if the **Posterior Expected Loss** is within our constraints.

1 | summary(t1) |

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 | ## Quantiles of posteriors for A and B: ## ## $Probability ## $Probability$A_probs ## 0% 25% 50% 75% 100% ## 0.06019053 0.08936888 0.09552425 0.10188340 0.14012802 ## ## $Probability$B_probs ## 0% 25% 50% 75% 100% ## 0.04926047 0.07582110 0.08158084 0.08758630 0.12639536 ## ## ## -------------------------------------------- ## ## P(A > B) by (0)%: ## ## $Probability ## [1] 0.86214 ## ## -------------------------------------------- ## ## Credible Interval on (A - B) / B for interval length(s) (0.9) : ## ## $Probability ## 5% 95% ## -0.07634565 0.48644269 ## ## -------------------------------------------- ## ## Posterior Expected Loss for choosing B over A: ## ## $Probability ## [1] 0.01270441 |

1 | summary(t2) |

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 | ## Quantiles of posteriors for A and B: ## ## $Probability ## $Probability$A_probs ## 0% 25% 50% 75% 100% ## 0.06244681 0.08940537 0.09554403 0.10195937 0.13931010 ## ## $Probability$B_probs ## 0% 25% 50% 75% 100% ## 0.007008903 0.017726756 0.020601621 0.023791156 0.050439822 ## ## ## -------------------------------------------- ## ## P(A > B) by (0)%: ## ## $Probability ## [1] 1 ## ## -------------------------------------------- ## ## Credible Interval on (A - B) / B for interval length(s) (0.9) : ## ## $Probability ## 5% 95% ## 2.173703 5.994963 ## ## -------------------------------------------- ## ## Posterior Expected Loss for choosing B over A: ## ## $Probability ## [1] NaN |

If the **Posterior Expected Loss** is lower than our threshold for caring on abs(A - B) then we can call this test and accept the current results. The PEL is small in both cases, and possibly 0/NaN for `t2`

so it’s quite clear that priors, even weak ones, have a significant positive effect on Type 1 Errors. Remember that we see this effect partially because our **priors** were of a similar shape to the **data**. If the **priors** and the **data** disagree, the effects might not be so clear cut and you will need more **data** to have a stable **posterior**.

Check out bayesAB 0.7.0 to supercharge your AB tests and reach out to me if you have any questions!

**leave a comment**for the author, please follow the link and comment on their blog:

**Frank Portman**.

R-bloggers.com offers

**daily e-mail updates**about R news and tutorials about learning R and many other topics. Click here if you're looking to post or find an R/data-science job.

Want to share your content on R-bloggers? click here if you have a blog, or here if you don't.