---
title: "Machine Learning in R with CivisML"
author: "Patrick Miller and Liz Sander"
date: "2018-1-18"
output: rmarkdown::html_vignette
vignette: >
%\VignetteIndexEntry{civis_ml}
%\VignetteEngine{knitr::rmarkdown}
%\VignetteEncoding{UTF-8}
---
There are so many models to build! When this becomes challenging on a local machine, offloading model building to the cloud can save a lot of time and effort.
[CivisML](https://medium.com/civis-analytics/civisml-scikit-learn-at-scale-b01b496916ea) is a machine learning service on Civis Platform that makes this as painless as possible. You can fit many different models, do extensive hyperparameter tuning, and score data sets with millions of observations stored in remote databases. Once these models are built, they live in Civis Platform permanently and can be included into production pipelines. Results can be easily incorporated into reports and dashboards.
CivisML is built in Python using [scikit-learn](http://scikit-learn.org/stable/), and leverages AWS behind the scenes for efficient distributed computing. However, most of its features can be used through R without knowledge of Python or AWS with the `civis_ml` function in `civis`.
While `civis_ml` is a complex function with many arguments, basic machine learning modeling and scoring can be easily carried out. We illustrate several features of `civis_ml` with data from a fictitious company called Brandable, who is looking to predict which customers are likely to upgrade from the free to the premium service.
## Data sources
The first step of modeling with `civis_ml` is to specify the data source, which is the first argument. `civis_ml` works with local data frames, a CSV on local disk, [feather-format](https://github.com/wesm/feather) files, tables in Redshift, and files on S3 (the files endpoint):
```{r, eval = FALSE}
library(civis)
civis_ml(df, ...)
civis_ml("path/to/data.csv", ...)
civis_ml(civis_table(table_name = "schema.table", database_name = "database"), ...)
civis_ml(civis_file(1234), ...)
```
The Brandable data is located in a Redshift table called `sample_project.premium_training_set`.
```{r, eval = FALSE}
options(civis.default_db = "my_database")
tab <- civis_table(table_name = "sample_project.premium_training_set")
```
Note that `civis_table` only returns information on where to find the data for `civis_ml`, not the data itself. `civis_table` also takes two SQL statements that can be useful for limiting the rows used for training: `sql_where`, and `sql_limit`.
## Modeling
After the data source is specified, we next choose the model type. There are 13 named CivisML models that can be called from `civis_ml`, 6 for classification and 7 for regression. The name of the model corresponds to the name of the estimator in scikit-learn. It can be given in the `model_type` argument of `civis_ml`, or called directly using a `civis_ml_*` function such as `civis_ml_sparse_logistic`.
| Name | R Workflow | Model Type | scikit-learn Documentation |
|------|:-----------|------------|-----------|------------------|
`sparse_logistic` | `civis_ml_sparse_logistic` | classification | [Logistic Regression](http://scikit-learn.org/stable/modules/generated/sklearn.linear_model.LogisticRegression.html) |
`gradient_boosting_classifier` | `civis_ml_gradient_boosting_classifier` | classification | [GradientBoostingClassifier](http://scikit-learn.org/stable/modules/generated/sklearn.ensemble.GradientBoostingClassifier.html) |
`random_forest_classifier` | `civis_ml_random_forest_classifier` | classification | [RandomForestClassifier](http://scikit-learn.org/stable/modules/generated/sklearn.ensemble.RandomForestClassifier.html) |
`extra_trees_classifier` | `civis_ml_extra_trees_classifier` | classification | [ExtraTreesClassifier](http://scikit-learn.org/stable/modules/generated/sklearn.ensemble.ExtraTreesClassifier.html)|
`multilayer_perceptron_classifier` | | classification | [muffnn.MLPClassifier](https://github.com/civisanalytics/muffnn) |
`stacking_classifier` | | classification | [StackedClassifier](https://github.com/civisanalytics/civisml-extensions)
`sparse_linear_regressor` | `civis_ml_sparse_linear_regressor` | regression | [LinearRegression](http://scikit-learn.org/stable/modules/generated/sklearn.linear_model.LinearRegression.html) |
`sparse_ridge_regressor` | `civis_ml_sparse_ridge_regressor` | regression | [Ridge](http://scikit-learn.org/stable/modules/generated/sklearn.linear_model.Ridge.html) |
`gradient_boosting_regressor` | `civis_ml_gradient_boosting_regressor` | regression | [GradientBoostingRegressor](http://scikit-learn.org/stable/modules/generated/sklearn.ensemble.GradientBoostingRegressor.html) |
`random_forest_regressor` | `civis_ml_random_forest_regressor` | regression | [RandomForestRegressor](http://scikit-learn.org/stable/modules/generated/sklearn.ensemble.RandomForestRegressor.html)|
`extra_trees_regressor` | `civis_ml_extra_trees_regressor` | regression | [ExtraTreesRegressor](http://scikit-learn.org/stable/modules/generated/sklearn.ensemble.ExtraTreesRegressor.html)|
`multilayer_perceptron_regressor` | | regression | [muffnn.MLPRegressor](https://github.com/civisanalytics/muffnn) |
`stacking_regressor` | | regression | [StackedRegressor](https://github.com/civisanalytics/civisml-extensions)
Documentation on the meta parameters specific to each estimator are provided in `?civis_ml_*`. For example, the regularization strength parameter `C` of `sparse_logistic` is documented in `?civis_ml_sparse_logistic`.
For the Brandable data, we use a `random_forest` classifier to predict the probability that a customer upgrades from free to premium services. For efficiency, we can also denote a `primary_key`, and a set of `excluded_columns` that are not included in the model:
```{r, eval = FALSE}
library(civis)
tab <- civis_table("sample_project.premium_training_set")
m <- civis_ml(tab, dependent_variable = "upgrade",
model_type = "random_forest_classifier",
primary_key = "brandable_user_id",
excluded_columns = "residential_zip")
m <- civis_ml_random_forest_classifier(tab,
primary_key = "brandable_user_id",
excluded_columns = "residential_zip")
```
Note that if the dependent variables have null values, those rows will be removed before modeling.
### Hyperparameter Tuning
You can tune hyperparameters using one of two methods: grid search or hyperband. CivisML will perform grid search if you pass a named list of hyperparameters and candidate values to `cross_validation_parameters`. By default, hyperparameter tuning will run in parallel, using as many jobs as possible without overloading your computing cluster. If you wish to have more control over the number of jobs running at once, you can set it using the `n_jobs` parameter.
[Hyperband](https://arxiv.org/abs/1603.06560) is an efficient approach to hyperparameter optimization, and recommended over grid search where possible. CivisML will perform hyperband optimization if you pass the string `"hyperband"` to `cross_validation_parameters`. Hyperband cannot be used to tune GLMs. For this reason, preset GLMs do not have a hyperband option. Hyperband is supported for random forests, gradient boosted trees, extra trees, multilayer perceptrons, and the random forest and gradient boosted tree steps of stacking. It is highly recommended that multilayer perceptron models only be used with hyperband.
For the `random_forest_classifier` in the Brandable data, we try both `"hyperband"` and grid search for hyperparameter optimization.
```{r, eval = FALSE}
tab <- civis_table("sample_project.premium_training_set")
# hyperband
m_hyper <- civis_ml(tab, dependent_variable = "upgrade",
model_type = "random_forest_classifier",
primary_key = "brandable_user_id",
excluded_columns = "residential_zip",
cross_validation_parameters = 'hyperband')
# grid search
cv_params <- list("max_depth" = c(2, 3, 5),
"n_estimators" = c(50, 100, 500))
m_grid <- civis_ml(tab, dependent_variable = "upgrade",
model_type = "random_forest_classifier",
primary_key = "brandable_user_id",
excluded_columns = "residential_zip",
cross_validation_parameters = cv_params)
```
CivisML runs pre-defined models with hyperband using the following distributions:
| Models | Cost Parameter | Hyperband Distributions |
| ---------------------------------- | ------------------ | --------------------------------------------------------------------------- |
| gradient_boosting_classifier
gradient_boosting_regressor
GBT step in stacking_classifier
GBT step in stacking_regressor | `n_estimators`
`min = 100,`
`max = 1000` | `max_depth: randint(low=1, high=5)`
`max_features: [None, 'sqrt', 'log2', 0.5, 0.3, 0.1, 0.05, 0.01]`
`learning_rate: truncexpon(b=5, loc=.0003, scale=1./167.)` |
| ---------------------------------- | ------------------ | --------------------------------------------------------------------------- |
| random_forest_classifier
random_forest_regressor
extra_trees_classifier
extra_trees_regressor
RF step in stacking_classifier
RF step in stacking_regressor | `n_estimators`
`min = 100,`
`max = 1000` | `criterion: ['gini', 'entropy']`
`max_features: truncexpon(b=10., loc=.01, scale=1./10.11)`
`max_depth: [1, 2, 3, 4, 6, 10, None]` |
| ---------------------------------- | ------------------ | --------------------------------------------------------------------------- |
| multilayer_perceptron_classifier
multilayer_perceptron_regressor | `n_epochs`
`min = 5,`
`max = 50` | `keep_prob: uniform()`
``hidden_units: [(), (16,), (32,), (64,), (64, 64), (64, 64, 64),`
`(128,), (128, 128), (128, 128, 128), (256,),`
`(256, 256), (256, 256, 256), (512, 256, 128, 64),`
`(1024, 512, 256, 128)]`
`learning_rate: [1e-2, 2e-2, 5e-2, 8e-2, 1e-3, 2e-3, 5e-3, 8e-3, 1e-4]` |
The truncated exponential distribution for the gradient boosting
classifier and regressor was chosen to skew the distribution toward
small values, ranging between .0003 and .03, with a mean close to
.006. Similarly, the truncated exponential distribution for the random
forest and extra trees models skews toward small values, ranging
between .01 and 1, and with a mean close to .1.
### Stacking
The `"stacking_classifier"` model stacks together the `"gradient_boosting_classifier"` and `"random_forest_classifier"` predefined models together with a
`glmnet.LogitNet(alpha=0, n_splits=4, max_iter=10000, tol=1e-5,
scoring='log_loss')`. Defaults for the predefined models are documented in `?civis_ml`. Each column is first [standardized](http://scikit-learn.org/stable/modules/generated/sklearn.preprocessing.StandardScaler.html), and then the model predictions are combined using [LogisticRegressionCV](http://scikit-learn.org/stable/modules/generated/sklearn.linear_model.LogisticRegressionCV.html) with `penalty='l2'` and `tol=1e-08`. The `"stacking_regressor"` works similarly, stacking together the `"gradient_boosting_regressor"` and `"random_forest_regressor"` models and a `glmnet.ElasticNet(alpha=0, n_splits=4, max_iter=10000,
tol=1e-5, scoring='r2')`, combining them using [NonNegativeLinearRegression](https://github.com/civisanalytics/civisml-extensions).
```{r, eval = FALSE}
m_stack <- civis_ml(tab, dependent_variable = "upgrade",
model_type = "stacking_classifier",
primary_key = "brandable_user_id",
excluded_columns = "residential_zip")
```
## Results
A simple summary of the results from the best fitting model is provided with `print`:
```{r, eval=FALSE}
m
```
```{r run_model, eval=FALSE, echo=FALSE}
# use this chunk to actually update the model if necessary
library(civis)
tab <- civis_table("sample_project.premium_training_set")
cv_params <- list("max_depth" = c(2, 3, 5),
"n_estimators" = c(50, 100, 500))
m <- civis_ml(tab, dependent_variable = "upgrade",
model_type = "random_forest_classifier",
primary_key = "brandable_user_id",
excluded_columns = "residential_zip",
cross_validation_parameters = cv_params)
saveRDS(m, file = "../inst/civis_ml_brandable.rds")
oos <- fetch_oos_scores(m)
saveRDS(oos, file = "../inst/civis_ml_oos.rds")
err_m <- tryCatch({
civis_ml(tab, dependent_variable = "upgrade",
model_type = "random_fest_classifier",
primary_key = "brandable_user_id",
excluded_columns = "residential_zip",
cross_validation_parameters = cv_params)
}, error = function(e) e)
saveRDS(err_m, file = "../inst/civis_ml_err.rds")
```
```{r, eval=TRUE, echo=FALSE}
library(civis)
path <- system.file("civis_ml_brandable.rds", package = 'civis')
m <- readRDS(path)
m
```
Following the link takes you to a summary of the model results in Civis Platform. Additional metrics can be computed with `get_metric`:
```{r}
get_metric(m, "accuracy")
get_metric(m, "confusion_matrix")
get_metric(m, "roc_auc")
```
Out of sample (or out of fold) scores used in training can be retrieved using `fetch_oos_scores`:
```{r, echo=TRUE, eval=FALSE}
oos <- fetch_oos_scores(m)
head(oos)
```
```{r, echo=FALSE, eval=TRUE}
path <- system.file("civis_ml_oos.rds", package = 'civis')
oos <- readRDS(path)
head(oos)
```
## Diagnostics
For classification problems, `plot` produces a a decile plot using `ggplot2`. For the premium upgrade model, the decile plot shows that the top-scoring 10\% of individuals contain 2.20 times as many targets (people who upgraded) as a randomly selected list of the same size.
```{r, fig.width = 5}
plot(m)
```
For regression problems, `plot` produces a binned scatter-plot of $y$ against $\hat{y}$.
`hist` shows the histogram of out of sample (out of fold scores), also using `ggplot2`:
```{r}
hist(m)
```
## Prediction and Scoring
CivisML can also be used to score models on hundreds of millions of rows, and distributed over many compute instances. Like many estimators in R, this is done through a `predict` method. The `newdata` argument of `predict` can take any data source supported in `civis_ml`. Here we use a table in Redshift containing all Brandable users, and output the result to another table in Redshift:
```{r, eval=FALSE}
pred_tab <- civis_table(table_name = "sample_project.brandable_all_users")
pred_job <- predict(m, newdata = pred_tab,
output_table = "sample_project.brandable_user_scores")
```
Like training and validation, scoring is distributed by default, using up to 90 percent of your computing cluster resources. If you would like to have more control over the number of jobs that are run at once, you can set a maximum using `n_jobs`:
```{r, eval=FALSE}
pred_job <- predict(m, newdata = pred_tab,
output_table = "sample_project.brandable_user_scores",
n_jobs = 25)
```
The predictions can be loaded into memory using `fetch_predictions`, which downloads directly from S3:
```{r, eval=FALSE}
yhat <- fetch_predictions(pred_job)
```
Note that if the table of predictions exceeds available memory, it may be helpful to use `download_civis` instead.
```{r, eval=FALSE}
# download from S3
download_civis(pred_job$model_info$output_file_ids, path = "my_predictions.csv")
# download from Redshift
download_civis("sample_project.brandable_user_scores")
```
## Retrieving Existing models
An existing model (or particular run of an existing model) can be retrieved using `civis_ml_fetch_existing`:
```{r, eval=FALSE}
model_id <- m$job$id
m <- civis_ml_fetch_existing(model_id)
```
## Error Handling
Unfortunately, many kinds of errors can occur. When an error occurs within CivisML, a `civis_ml_error` is thrown. By default, the log from the CivisML job is printed, which is useful for debugging.
Here is an example error from misspelling the model type:
```{r, eval=FALSE}
civis_ml(tab, dependent_variable = "upgrade",
model_type = "random_fest_classifier",
primary_key = "brandable_user_id",
excluded_columns = "residential_zip",
cross_validation_parameters = cv_params)
```
```{r, echo=FALSE, eval=TRUE}
path <- system.file("civis_ml_err.rds", package = 'civis')
err <- readRDS(path)
err
```
If you don't understand the error message, providing the error message, job, and run ids to support is the best way to get help!
## Programming with `civis_ml`
When programming with `civis_ml`, errors can be caught using the base R `try` or `tryCatch`. In `civis`, we provide functions for getting debugging information using `get_error` or just the logs using `fetch_logs`.
```{r, eval = FALSE}
e <- tryCatch({
civis_ml(tab, dependent_variable = "upgrade",
model_type = "random_fest_classifier",
primary_key = "brandable_user_id",
excluded_columns = "residential_zip")
}, civis_ml_error = function(e) e)
get_error(e)
fetch_logs(e)
```
Error handling can be used to implement more robust workflow programming with `civis_ml`. In the following function, we implement `retry_model`, which retries on e.g. connection failures but not on a `civis_ml_error`.
```{r, eval=FALSE}
retry_model <- function(max_retries = 5) {
i <- 1
while (i < max_retries) {
tryCatch({
m <- civis_ml(tab, dependent_variable = "upgrade",
model_type = "random_forest_classifier",
primary_key = "brandable_user_id",
excluded_columns = "residential_zip")
return(m)
}, civis_ml_error = function(e) stop(e))
cat("Retry: ", i, fill = TRUE)
i <- i + 1
}
stop("Exceeded maximum retries.")
}
```
Workflow programming could be further enhanced by printing the logs, storing the error object, or writing error logs to a file or database.
## Appendix
### Parallelization
To fit many models in parallel using `parallel`, `foreach`, or `future`, check out [this article](https://civisanalytics.github.io/civis-r/articles/concurrency.html) or the vignette on concurrency at `browseVignettes("civis")`.
### Sample weights
Many estimators take a `sample_weight` argument. This can be be specified with the `fit_params` argument of `civis_ml` using `list(sample_weight = 'survey_weight_column')`.
### Missing data
Modeling data must be complete. Any missing values will be imputed with the mean of non-null values in a column.
### CivisML Versions
By default, CivisML uses its latest version in production.
If you would like a specific version
(e.g., for a production pipeline where pinning the CivisML version is desirable),
both `civis_ml` and the `civis_ml_*` functions have the optional parameter
``civisml_version`` that accepts a string, e.g., ``'v2.3'``
for CivisML v2.3. Please see [here](https://civis.zendesk.com/hc/en-us/articles/360000260011-CivisML) for the list of CivisML versions.
### More information
Custom estimators can be written in Python and included in CivisML if they follow the scikit-learn API. For example, the `sparse_logistic`, `sparse_linear_regressor`, and `sparse_ridge_regressor` models all use the public Civis Analytics [glmnet](https://github.com/civisanalytics/python-glmnet) wrapper in Python.
Browse [the CivisML documentation](https://civis-python.readthedocs.io/en/stable/ml.html) for more details!