The goal of insurancerating
is to give analytic techniques that can be used in insurance rating. It helps actuaries to implement GLMs within all relevant steps needed to construct a risk premium from raw data.
It provides a data driven strategy for the construction of tariff classes in P&C insurance. The goal is to bin the continuous factors such that categorical risk factors result which capture the effect of the covariate on the response in an accurate way, while being easy to use in a generalized linear model (GLM).
insurancerating
also provides recipes on how to easily perform univariate analyses on an insurance portfolio. In addition it adds functionality to include reference categories in the levels of the coefficients in the output of a generalized linear regression analysis.
Install insurancerating from CRAN:
install.packages("insurancerating")
Or the development version from GitHub:
# install.packages("remotes")
remotes::install_github("MHaringa/insurancerating")
This is a basic example which shows the techniques provided in insurancerating.
The first part shows how to fit a GAM for the variable age_policyholder in the MTPL dataset:
library(insurancerating)
# Claim frequency
age_policyholder_frequency <- fit_gam(data = MTPL,
nclaims = nclaims,
x = age_policyholder,
exposure = exposure)
# Claim severity
age_policyholder_severity <- fit_gam(data = MTPL,
nclaims = nclaims,
x = age_policyholder,
exposure = exposure,
amount = amount,
model = "severity")
Create plot:
autoplot(age_policyholder_frequency, show_observations = TRUE)
Determine classes for the claim frequency (the points show the ratio between the observed number of claims and exposure for each age). This method is based on the work by Henckaerts et al. (2018), see ?construct_tariff_classes
for the reference.
clusters_freq <- construct_tariff_classes(age_policyholder_frequency)
clusters_sev <- construct_tariff_classes(age_policyholder_severity)
autoplot(clusters_freq, show_observations = TRUE)
In this example the term exposure is a measure of what is being insured. Here an insured vehicle is an exposure. If the vehicle is insured as of July 1 for a certain year, then during that year, this would represent an exposure of 0.5 to the insurance company.
The figure shows that younger policyholders have a higher risk profile. The fitted GAM is lower than might be expected from the observed claim frequency for policyholders of age 19. This is because there are very few young policyholders of age 19 present in the portfolio.
The GAM for the claim severity :
autoplot(age_policyholder_severity,
show_observations = TRUE,
remove_outliers = 100000)
The second part adds the constructed tariff classes for the variable age_policyholder
to the dataset, and sets the base level of the factor age_policyholder
to the level with the largest exposure. In this example for claim frequency the class for ages (39,50], which contains the largest exposure.
library(dplyr)
dat <- MTPL |>
mutate(age_policyholder_freq_cat = clusters_freq$tariff_classes) |>
mutate(across(where(is.character), as.factor)) |>
mutate(across(where(is.factor), ~biggest_reference(., exposure)))
glimpse(dat)
## Rows: 30,000
## Columns: 8
## $ age_policyholder <int> 70, 40, 78, 49, 59, 71, 55, 52, 51, 47, 62, …
## $ nclaims <int> 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0,…
## $ exposure <dbl> 1.0000000, 1.0000000, 1.0000000, 1.0000000, …
## $ amount <dbl> 0, 0, 0, 0, 0, 0, 2607, 0, 0, 0, 0, 2890, 0,…
## $ power <int> 106, 74, 65, 64, 29, 66, 43, 55, 100, 66, 44…
## $ bm <int> 5, 3, 8, 10, 1, 6, 2, 2, 1, 2, 1, 2, 2, 9, 5…
## $ zip <fct> 1, 1, 2, 1, 3, 3, 3, 3, 3, 3, 2, 1, 1, 2, 2,…
## $ age_policyholder_freq_cat <fct> "(39,84]", "(39,84]", "(39,84]", "(39,84]", …
The last part is to fit a generalized linear model. rating_factors()
prints the output including the reference group.
model_freq1 <- glm(nclaims ~ age_policyholder_freq_cat,
offset = log(exposure),
family = "poisson",
data = dat)
model_freq2 <- glm(nclaims ~ age_policyholder_freq_cat + age_policyholder,
offset = log(exposure),
family = "poisson",
data = dat)
x <- rating_factors(model_freq1, model_freq2)
x
## risk_factor level est_model_freq1 est_model_freq2
## 1 (Intercept) (Intercept) 0.1179256 0.2458900
## 2 age_policyholder_freq_cat (39,84] 1.0000000 1.0000000
## 3 age_policyholder_freq_cat [18,25] 2.2168483 1.4530987
## 4 age_policyholder_freq_cat (25,32] 1.5176253 1.0739519
## 5 age_policyholder_freq_cat (32,39] 1.2278612 0.9526827
## 6 age_policyholder_freq_cat (84,95] 0.5887492 0.9016893
## 7 age_policyholder age_policyholder NA 0.9867281
autoplot.riskfactor()
creates a figure. The base level of the factor age_policyholder_freq_cat
is the group with the largest exposure and is shown first.
autoplot(x)
Include model_data
to sort the clustering in the original order. Ordering the factor age_policyholder_freq_cat
only works when biggest_reference()
is used to set the base level of the factor to the level with the largest exposure.
rf <- rating_factors(model_freq1, model_freq2,
model_data = dat)
autoplot(rf)
The following graph includes the exposure as a bar graph and shows some more options:
rf <- rating_factors(model_freq1, model_freq2,
model_data = dat, exposure = exposure)
autoplot(rf, linetype = TRUE)
Add predictions to the data set:
dat_pred <- dat |>
add_prediction(model_freq1, model_freq2)
glimpse(dat_pred)
## Rows: 30,000
## Columns: 10
## $ age_policyholder <int> 70, 40, 78, 49, 59, 71, 55, 52, 51, 47, 62, …
## $ nclaims <int> 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0,…
## $ exposure <dbl> 1.0000000, 1.0000000, 1.0000000, 1.0000000, …
## $ amount <dbl> 0, 0, 0, 0, 0, 0, 2607, 0, 0, 0, 0, 2890, 0,…
## $ power <int> 106, 74, 65, 64, 29, 66, 43, 55, 100, 66, 44…
## $ bm <int> 5, 3, 8, 10, 1, 6, 2, 2, 1, 2, 1, 2, 2, 9, 5…
## $ zip <fct> 1, 1, 2, 1, 3, 3, 3, 3, 3, 3, 2, 1, 1, 2, 2,…
## $ age_policyholder_freq_cat <fct> "(39,84]", "(39,84]", "(39,84]", "(39,84]", …
## $ pred_nclaims_model_freq1 <dbl> 0.11792558, 0.11792558, 0.11792558, 0.117925…
## $ pred_nclaims_model_freq2 <dbl> 0.09650865, 0.14409239, 0.08672539, 0.127766…
Compute indices of model performance for GLMs. The RMSE is the square root of the average of squared differences between prediction and actual observation and indicates the absolute fit of the model to the data. It can be interpreted as the standard deviation of the unexplained variance, and is in the same units as the response variable.
model_performance(model_freq1, model_freq2)
## [34m# Comparison of Model Performance Indices
##
## [39mModel | AIC | BIC | RMSE
## ----------------------------------------
## model_freq1 | 22983.34 | 23024.88 | 0.36
## model_freq2 | 22943.06 | 22992.92 | 0.36
To test the stability of the predictive ability of the fitted model it might be helpful to determine the variation in the computed RMSE. The variation is calculated by computing the root mean squared errors from generated bootstrap replicates.
For claim severity models it might be helpful to test the variation in the RMSE in case the portfolio contains large claim sizes. The figure below shows that the variation in the RMSE of the frequency model is quite low (as expected). The dashed line shows the RMSE of the fitted (original) model, the other lines represent the 95% confidence interval.
bootstrap_rmse(model_freq1, dat, n = 100, show_progress = FALSE) |>
autoplot()
Check Poisson GLM for overdispersion. A dispersion ratio larger than one indicates overdispersion, this occurs when the observed variance is higher than the variance of the theoretical model. If the dispersion ratio is close to one, a Poisson model fits well to the data. A p-value < .05 indicates overdispersion. Overdispersion > 2 probably means there is a larger problem with the data: check (again) for outliers.
check_overdispersion(model_freq1)
## dispersion ratio = 1.185
## Pearson's Chi-Squared = 35554.163
## p-value = < 0.001
## Overdispersion detected.
Misspecifications in GLMs cannot reliably be diagnosed with standard residual plots, and GLMs are thus often not as thoroughly checked as LMs. One reason why GLMs residuals are harder to interpret is that the expected distribution of the data changes with the fitted values. As a result, standard residual plots, when interpreted in the same way as for linear models, seem to show all kind of problems, such as non-normality, heteroscedasticity, even if the model is correctly specified. check_residuals()
aims at solving these problems by creating readily interpretable residuals for GLMs that are standardized to values between 0 and 1, and that can be interpreted as intuitively as residuals for the linear model. This is achieved by a simulation-based approach, similar to the Bayesian p-value or the parametric bootstrap, that transforms the residuals to a standardized scale. This explanation is adopted from the vignette for DHARMa.
Detect deviations from the expected distribution, and produce a uniform quantile-quantile plot. The simulated residuals in the QQ plot below show no clear deviation from a Poisson distribution. Note that formal tests almost always yield significant results for the distribution of residuals and visual inspections (e.g. Q-Q plots) are preferable.
check_residuals(model_freq1, n_simulations = 1000) |>
autoplot()
It might happen that in the fitted model for a data point all simulations have the same value (e.g. zero), this returns the error message Error in approxfun: need at least two non-NA values to interpolate. If that is the case, it could help to increase the number of simulations (e.g. n = 1000).
This is a basic example which shows how to easily perform an univariate analysis on a MTPL portfolio using insurancerating
.
An univariate analysis consists in the evaluation of overall claim frequency, severity and risk premium. Its main purpose lies in verifying the experience data reasonableness using previous experience comparison and professional judgement.
univariate()
shows the basic risk indicators split by the levels of the discrete risk factor:
library(insurancerating)
univariate(MTPL2,
x = area, # discrete risk factor
nclaims = nclaims, # number of claims
exposure = exposure,
premium = premium,
severity = amount) # loss
## area amount nclaims exposure premium frequency average_severity
## <int> <int> <int> <num> <int> <num> <num>
## 1: 2 4063270 98 818.53973 51896 0.11972540 41461.94
## 2: 3 7945311 113 764.99178 49337 0.14771401 70312.49
## 3: 1 6896187 146 1065.74795 65753 0.13699299 47234.16
## 4: 0 6922 1 13.30685 902 0.07514927 6922.00
## risk_premium loss_ratio average_premium
## <num> <num> <num>
## 1: 4964.0474 78.296400 63.40071
## 2: 10386.1390 161.041632 64.49350
## 3: 6470.7486 104.880188 61.69658
## 4: 520.1832 7.674058 67.78464
The following indicators are calculated:
Here the term exposure is a measure of what is being insured. For example, an insured vehicle is an exposure. If the vehicle is insured as of July 1 for a certain year, then during that year, this would represent an exposure of 0.5 to the insurance company. The term risk premium is used here as an equivalent of pure premium and burning cost.
univariate()
ignores missing input arguments, for instance only the claim frequency is calculated when premium
and severity
are unknown:
univariate(MTPL2, x = area, nclaims = nclaims, exposure = exposure)
## area nclaims exposure frequency
## <int> <int> <num> <num>
## 1: 2 98 818.53973 0.11972540
## 2: 3 113 764.99178 0.14771401
## 3: 1 146 1065.74795 0.13699299
## 4: 0 1 13.30685 0.07514927
However, the above table is small and easy to understand, the same information might be presented more effectively with a graph, as shown below.
univariate(MTPL2, x = area, nclaims = nclaims, exposure = exposure) |>
autoplot()
In autoplot.univariate()
, show_plots
defines the plots to show and also the order of the plots. The following plots are available:
For example, to show the exposure and claim frequency plots:
un <- univariate(MTPL2, x = area, nclaims = nclaims, exposure = exposure)
autoplot(un, show_plots = c(6,1))
To check whether claim frequency is consistent over the years:
df <- MTPL2 |>
mutate(year = sample(2016:2019, nrow(MTPL2), replace = TRUE))
un <- univariate(df, x = area, nclaims = nclaims,
exposure = exposure, by = year)
autoplot(un, show_plots = 1)
To remove the bars from the plot with the line graph, add background = FALSE
:
univariate(MTPL2, x = area, nclaims = nclaims, exposure = exposure) |>
autoplot(show_plots = c(6,1), background = FALSE)
sort
orders the levels of the risk factor into descending order by exposure:
univariate(MTPL2, x = area, nclaims = nclaims, exposure = exposure) |>
autoplot(show_plots = c(6,1), background = FALSE, sort = TRUE)
sort_manual
in autoplot.univariate()
can be used to sort the levels of the discrete risk factor into your own ordering. This makes sense when the levels of the risk factor has a natural order, or when not all levels of the risk factor are desired in the output.
univariate(MTPL2, x = area, nclaims = nclaims, exposure = exposure) |>
autoplot(show_plots = c(6,1), background = FALSE,
sort_manual = c("2", "3", "1", "0"))
The following graph shows some more options:
univariate(MTPL2, x = area, nclaims = nclaims, exposure = exposure) |>
autoplot(show_plots = c(6,1), background = FALSE, sort = TRUE, ncol = 2,
color_bg = "dodgerblue", color = "blue",
custom_theme = ggplot2::theme_classic())
Or create a bar graph for the number of claims:
univariate(MTPL2, x = area, nclaims = nclaims) |>
autoplot(show_plots = 8, coord_flip = TRUE, sort = TRUE)
For continuous variables a histogram can be created:
histbin(MTPL2, premium)
Two ways of displaying numerical data over a very wide range of values in a compact way are taking the logarithm of the variable, or omitting the outliers. Both do not show the original distribution, however. Another way is to create one bin for all the outliers. This yields both the original distribution, and also gives a feel for the number of outliers.
histbin(MTPL2, premium, right = 110)
This is a basic example which shows how to easily perform model refinement using insurancerating
. insurancerating
can be used to impose either smoothing to the parameter estimates or to add restrictions to the parameter estimates. These methods are deduced from the article Third Party Motor Liability Ratemaking with R, by Spedicato, G. (2012).
Fit (again) a Poisson GLM and a Gamma GLM, and combine them to determine premiums:
mod_freq <- glm(nclaims ~ zip + age_policyholder_freq_cat,
offset = log(exposure),
family = "poisson",
data = dat)
mod_sev <- glm(amount ~ bm + zip,
weights = nclaims,
family = Gamma(link = "log"),
data = dat %>% filter(amount > 0))
MTPL_premium <- dat |>
add_prediction(mod_freq, mod_sev) |>
mutate(premium = pred_nclaims_mod_freq * pred_amount_mod_sev)
Fit a burning model without restrictions. Even though restrictions could be applied to frequency and severity models, it is more appropriate to add restrictions (and smoothing) to the risk premium model.
burn_unrestricted <- glm(premium ~ zip + bm + age_policyholder_freq_cat,
weights = exposure,
family = Gamma(link = "log"),
data = MTPL_premium)
Smoothing can be used to reduce the tolerance for rate change. In smooth_coef()
, x_cut
is the name of the risk factor with clusters, x_org
is the name of the original risk factor, degree
is the order of the polynomial, and breaks
is a numerical vector with new clusters for x_org
. The smoothed estimates are added as an offset term to the model. An offset is just a fixed term added to the linear predictor, therefore if there is already an offset in the model, the offset terms are added together first (i.e. offset = log (a) + log (b) = log (a⋅b)).
burn_unrestricted |>
smooth_coef(x_cut = "age_policyholder_freq_cat",
x_org = "age_policyholder",
breaks = seq(18, 95, 5)) |>
print()
autoplot()
creates a figure for the smoothed estimates. The blue segments show the estimates from the unrestricted model. The red segments are the new estimates based on the polynomial.
burn_unrestricted |>
smooth_coef(x_cut = "age_policyholder_freq_cat",
x_org = "age_policyholder",
breaks = seq(18, 95, 5)) |>
autoplot()
degree
can be used to change the order of the polynomial:
burn_unrestricted |>
smooth_coef(x_cut = "age_policyholder_freq_cat",
x_org = "age_policyholder",
degree = 1,
breaks = seq(18, 95, 5)) |>
autoplot()
smooth_coef()
must always be followed by update_glm()
to refit the GLM.
burn_restricted <- burn_unrestricted |>
smooth_coef(x_cut = "age_policyholder_freq_cat",
x_org = "age_policyholder",
breaks = seq(18, 95, 5)) |>
update_glm()
# Show rating factors
rating_factors(burn_restricted)
## risk_factor level est_burn_restricted
## 1 (Intercept) (Intercept) 9184.0425714
## 2 zip 1 1.0000000
## 3 zip 0 0.3540330
## 4 zip 2 0.7311169
## 5 zip 3 0.7537306
## 6 age_policyholder_smooth [18,23] 2.3111217
## 7 age_policyholder_smooth (23,28] 1.7196059
## 8 age_policyholder_smooth (28,33] 1.3774398
## 9 age_policyholder_smooth (33,38] 1.2054716
## 10 age_policyholder_smooth (38,43] 1.1372518
## 11 age_policyholder_smooth (43,48] 1.1190324
## 12 age_policyholder_smooth (48,53] 1.1097671
## 13 age_policyholder_smooth (53,58] 1.0811117
## 14 age_policyholder_smooth (58,63] 1.0174238
## 15 age_policyholder_smooth (63,68] 0.9157627
## 16 age_policyholder_smooth (68,73] 0.7858894
## 17 age_policyholder_smooth (73,78] 0.6502670
## 18 age_policyholder_smooth (78,83] 0.5440602
## 19 age_policyholder_smooth (83,88] 0.5151356
## 20 age_policyholder_smooth (88,93] 0.6240618
## 21 bm bm 1.0370617
Most insurers have some form of a Bonus-Malus System in vehicle third party liability insurance. restrict_coef()
can be used to impose such restrictions. restrictions
must be a data.frame with in the first column the name of the column for which the restrictions should be applied and in the second column the restricted coefficients. The following example shows restrictions on the risk factor for region zip
:
zip_df <- data.frame(zip = c(0,1,2,3),
zip_restricted = c(0.8, 0.9, 1, 1.2))
burn_unrestricted |>
restrict_coef(restrictions = zip_df) |>
print()
To adjust the glm, restict_coef()
must always be followed by update_glm()
:
burn_restricted2 <- burn_unrestricted |>
restrict_coef(restrictions = zip_df) |>
update_glm()
rating_factors(burn_restricted2)
## risk_factor level est_burn_restricted2
## 1 (Intercept) (Intercept) 7766.5143894
## 2 age_policyholder_freq_cat (39,84] 1.0000000
## 3 age_policyholder_freq_cat [18,25] 2.1724584
## 4 age_policyholder_freq_cat (25,32] 1.4914700
## 5 age_policyholder_freq_cat (32,39] 1.2099529
## 6 age_policyholder_freq_cat (84,95] 0.5744446
## 7 zip_restricted 0 0.8000000
## 8 zip_restricted 1 0.9000000
## 9 zip_restricted 2 1.0000000
## 10 zip_restricted 3 1.2000000
## 11 bm bm 1.0371671
autoplot()
compares the restricted and the unrestricted estimates:
burn_unrestricted |>
restrict_coef(restrictions = zip_df) |>
autoplot()
burn_restricted3
combines restrict_coef()
and smooth_coef()
:
burn_restricted3 <- burn_unrestricted |>
restrict_coef(restrictions = zip_df) |>
smooth_coef(x_cut = "age_policyholder_freq_cat",
x_org = "age_policyholder",
breaks = seq(18, 95, 5)) |>
update_glm()
# Show rating factors
rating_factors(burn_restricted3)
## risk_factor level est_burn_restricted3
## 1 (Intercept) (Intercept) 7880.1570478
## 2 zip_restricted 0 0.8000000
## 3 zip_restricted 1 0.9000000
## 4 zip_restricted 2 1.0000000
## 5 zip_restricted 3 1.2000000
## 6 age_policyholder_smooth [18,23] 2.3111217
## 7 age_policyholder_smooth (23,28] 1.7196059
## 8 age_policyholder_smooth (28,33] 1.3774398
## 9 age_policyholder_smooth (33,38] 1.2054716
## 10 age_policyholder_smooth (38,43] 1.1372518
## 11 age_policyholder_smooth (43,48] 1.1190324
## 12 age_policyholder_smooth (48,53] 1.1097671
## 13 age_policyholder_smooth (53,58] 1.0811117
## 14 age_policyholder_smooth (58,63] 1.0174238
## 15 age_policyholder_smooth (63,68] 0.9157627
## 16 age_policyholder_smooth (68,73] 0.7858894
## 17 age_policyholder_smooth (73,78] 0.6502670
## 18 age_policyholder_smooth (78,83] 0.5440602
## 19 age_policyholder_smooth (83,88] 0.5151356
## 20 age_policyholder_smooth (88,93] 0.6240618
## 21 bm bm 1.0369179
And add refined premiums to the data:
premiums3 <- model_data(burn_restricted3) |>
add_prediction(burn_restricted3)
head(premiums3)
## age_policyholder age_policyholder_freq_cat_smooth age_policyholder_smooth
## <int> <num> <char>
## 1: 18 2.311122 [18,23]
## 2: 18 2.311122 [18,23]
## 3: 18 2.311122 [18,23]
## 4: 18 2.311122 [18,23]
## 5: 19 2.311122 [18,23]
## 6: 19 2.311122 [18,23]
## nclaims exposure amount power bm zip age_policyholder_freq_cat
## <int> <num> <num> <int> <int> <fctr> <fctr>
## 1: 1 1.00000000 261777 40 3 3 [18,25]
## 2: 0 0.09589041 0 68 5 2 [18,25]
## 3: 0 0.18630137 0 37 3 2 [18,25]
## 4: 0 0.18904110 0 33 1 2 [18,25]
## 5: 0 1.00000000 0 47 6 3 [18,25]
## 6: 1 0.06849315 6642 68 1 3 [18,25]
## pred_nclaims_mod_freq pred_amount_mod_sev premium zip_restricted
## <num> <num> <num> <num>
## 1: 0.26058660 67176.26 17505.233 1.2
## 2: 0.02353816 74215.89 1746.905 1.0
## 3: 0.04573127 68988.66 3154.939 1.0
## 4: 0.04640379 64129.60 2975.857 1.0
## 5: 0.26058660 74953.96 19531.999 1.2
## 6: 0.01784840 62444.85 1114.540 1.2
## pred_premium_burn_restricted3
## <num>
## 1: 24365.31
## 2: 21831.29
## 3: 20304.43
## 4: 18884.35
## 5: 27164.71
## 6: 22661.22