R/riskFormulations.R

Defines functions A_rc_over_b_vs_theta g_rc_over_b_vs_theta R_grad_rc_over_b_vs_theta R_rc_over_b_vs_theta A_rc_vs_theta g_rc_vs_theta R_grad_rc_vs_theta R_rc_vs_theta A_rc_vs_b_times_var R_grad_rc_vs_b_times_var R_rc_vs_b_times_var g_rc_vs_b_times_var A_rc_over_sd_vs_b_times_sd R_grad_rc_over_sd_vs_b_times_sd R_rc_over_sd_vs_b_times_sd g_rc_over_sd_vs_b_times_sd R_grad_rc_over_var R_rc_over_var g_rc_over_var A_rc_over_var_vs_b R_grad_rc_over_var_vs_b R_rc_over_var_vs_b g_rc_over_var_vs_b A_rc_over_b_double_index g_rc_over_b_double_index R_grad_rc_over_b_double_index R_rc_over_b_double_index A_rc_double_index g_rc_double_index R_grad_rc_double_index R_rc_double_index

# This file is meant to have the implementation of various risk concentration
# functions and its quantities such as gradient, Jacobian, etc, used by the
# successive convex approximation algorithm and by a general solver available
# in the riskParityPortfolio package.
# We use the following definitions of functions in this file:
#   - functions starting with the letter g are assumed to compute the risk
#     concentration vector.
#   - functions starting with the letter R followed by the name of the
#     formulation are assumed to compute the risk concentration. Most of them
#     are computed as sum(g ^ 2) unless a more efficient way is available.
#   - functions starting with "R_grad" are assumed to compute the gradient of
#     the risk concentration wrt the portfolio weights.
#   - functions starting with the letter A are assumed to compute the Jacobian
#     of the risk concentration vector wrt the portfolio weights.

#############################################################
# Compute g, R, and A for the formulation "rc-double-index" #
#############################################################

R_rc_double_index <- function(w, Sigma, b = NA, r = w*(Sigma %*% w)) {
  N <- length(w)
  return(2*(N*sum(r^2) - sum(r)^2))
}

R_grad_rc_double_index <- function(w, Sigma, b = NA, Sigma_w = Sigma %*% w, r = w*Sigma_w) {
  N <- length(w)
  v <- 4*(N*r - sum(r))
  return(as.vector(Sigma %*% (w*v) + Sigma_w*v))
}

g_rc_double_index <- function(w, Sigma, b = NA, r = w*(Sigma %*% w)) {
  N <- length(w)
  return(rep(r, times = N) - rep(r, each = N))
}

A_rc_double_index <- function(w, Sigma, b = NA, Sigma_w = Sigma %*% w) {
  N <- length(w)
  Sigma_w <- as.vector(Sigma_w)
  Ut <- diag(Sigma_w) + Sigma*w
  return(matrix(rep(t(Ut), N), ncol = N, byrow = TRUE) - matrix(rep(Ut, each = N), ncol = N))
}


####################################################################
# Compute g, R, and A for the formulation "rc-over-b-double-index" #
####################################################################

R_rc_over_b_double_index <- function(w, Sigma, b, r = w*(Sigma %*% w)) {
  N <- length(w)
  rb <- r/b
  return(2*(N*sum(rb^2) - sum(rb)^2))
}

R_grad_rc_over_b_double_index <- function(w, Sigma, b, Sigma_w = Sigma %*% w, r = w*Sigma_w) {
  N <- length(w)
  rb <- r/b
  v <- 4*(N*rb - sum(rb))/b
  return(as.vector(Sigma %*% (w*v) + Sigma_w*v))
}

g_rc_over_b_double_index <- function(w, Sigma, b, r = w*(Sigma %*% w)) {
  N <- length(w)
  rb <- r/b
  return(rep(rb, times = N) - rep(rb, each = N))
}

A_rc_over_b_double_index <- function(w, Sigma, b, Sigma_w = Sigma %*% w) {
  N <- length(w)
  Sigma_w <- as.vector(Sigma_w)
  Ut <- diag(Sigma_w) + Sigma*w
  Utb <- Ut / b
  return(matrix(rep(t(Utb), N), ncol = N, byrow = TRUE) - matrix(rep(Utb, each = N), ncol = N))
}


##############################################################
# Compute g, R, and A for the formulation "rc-over-var vs b" #
##############################################################

g_rc_over_var_vs_b <- function(w, Sigma, b, r = w*(Sigma %*% w)) {
  return(as.vector(r/sum(r) - b))
}

R_rc_over_var_vs_b <- function(w, Sigma, b, r = w*(Sigma %*% w)) {
  return(sum((g_rc_over_var_vs_b(w, Sigma, b, r = r))^2))
}

R_grad_rc_over_var_vs_b <- function(w, Sigma, b, Sigma_w = Sigma %*% w, r = w*Sigma_w) {
  sum_r <- sum(r)
  r_sumr_b <- r/sum_r - b
  v <- (2/sum_r)*(r_sumr_b - sum(r_sumr_b*r)/sum_r)
  return(as.vector(Sigma %*% (w*v) + Sigma_w*v))
}

A_rc_over_var_vs_b <- function(w, Sigma, b = NA, Sigma_w = Sigma %*% w, r = w*Sigma_w) {
  sum_r <- sum(r)
  Sigma_w <- as.vector(Sigma_w)
  r <- as.vector(r)
  Ut <- diag(Sigma_w) + Sigma*w
  return(Ut/sum_r - 2/(sum_r^2) * r %o% Sigma_w)
}


#########################################################
# Compute g, R, and A for the formulation "rc-over-var" #
#########################################################

g_rc_over_var <- function(w, Sigma, b = NA, r = w*(Sigma %*% w)) {
  return(as.vector(r/sum(r)))
}

R_rc_over_var <- function(w, Sigma, b = NA, r = w*(Sigma %*% w)) {
  return(sum((g_rc_over_var(w, Sigma, r = r))^2))
}

R_grad_rc_over_var <- function(w, Sigma, b, Sigma_w = Sigma %*% w, r = w*Sigma_w) {
  sum_r <- sum(r)
  r_sumr <- r/sum_r
  v <- (2/sum_r)*(r_sumr - sum(r_sumr^2))
  return(as.vector(Sigma %*% (w*v) + Sigma_w*v))
}

A_rc_over_var <- A_rc_over_var_vs_b


######################################################################
# Compute g, R, and A for the formulation "rc-over-sd vs b-times-sd" #
######################################################################

g_rc_over_sd_vs_b_times_sd <- function(w, Sigma, b, r = w*(Sigma %*% w)) {
  sqrt_sum_r <- sqrt(sum(r))
  return(r/sqrt_sum_r - b*sqrt_sum_r)
}

R_rc_over_sd_vs_b_times_sd <- function(w, Sigma, b, r = w*(Sigma %*% w)) {
  return(sum((g_rc_over_sd_vs_b_times_sd(w, Sigma, b, r = r))^2))
}

R_grad_rc_over_sd_vs_b_times_sd <- function(w, Sigma, b, Sigma_w = Sigma %*% w, r = w*Sigma_w) {
  sum_r <- sum(r)
  r_sumr_b <- r/sum_r - b
  v <- 2*r_sumr_b - sum(r_sumr_b^2)
  return(as.vector(Sigma %*% (w*v) + Sigma_w*v))
}

A_rc_over_sd_vs_b_times_sd <- function(w, Sigma, b, Sigma_w = Sigma %*% w, r = w*Sigma_w) {
  sum_r <- sum(r)
  Sigma_w <- as.vector(Sigma_w)
  r <- as.vector(r)
  Ut <- diag(Sigma_w) + Sigma * w
  return((Ut - (r/sum_r + b) %o% Sigma_w) / sqrt(sum_r))
}


###############################################################
# Compute g, R, and A for the formulation "rc vs b-times-var" #
###############################################################

g_rc_vs_b_times_var <- function(w, Sigma, b, r = w*(Sigma %*% w)) {
  return(as.vector(r - b*sum(r)))
}

R_rc_vs_b_times_var <- function(w, Sigma, b, r = w*(Sigma %*% w)) {
  return(sum((g_rc_vs_b_times_var(w, Sigma, b, r = r))^2))
}

R_grad_rc_vs_b_times_var <- function(w, Sigma, b, Sigma_w = Sigma %*% w, r = w*Sigma_w) {
  sum_r <- sum(r)
  v <- 2*(r - b*sum_r - sum(b*r) + sum(b^2)*sum_r)
  return(as.vector(Sigma %*% (w*v) + Sigma_w*v))
}

A_rc_vs_b_times_var <- function(w, Sigma, b, Sigma_w = Sigma %*% w, r = w*Sigma_w) {
  Sigma_w <- as.vector(Sigma_w)
  Ut <- diag(Sigma_w) + Sigma * w
  return(Ut - 2 * b %o% Sigma_w)
}


#########################################################
# Compute g, R, and A for the formulation "rc vs theta" #
#########################################################

R_rc_vs_theta <- function(wtheta, Sigma, b = NA, r = NA) {
  return(sum(g_rc_vs_theta(wtheta, Sigma)^2))
}

R_grad_rc_vs_theta <- function(wtheta, Sigma, b = NA) {
  N <- length(wtheta)-1
  w <- wtheta[1:N]
  theta <- wtheta[N+1]
  Sigma_w <- as.vector(Sigma %*% w)
  r <- as.vector(w*Sigma_w)
  v <- 2*(r - theta)
  return(c(as.vector(Sigma %*% (w*v) + Sigma_w*v), -sum(v)))
}

g_rc_vs_theta <- function(wtheta, Sigma, b = NA, r = NA) {
  N <- length(wtheta)-1
  theta <- wtheta[N+1]
  w <- wtheta[1:N]
  r <- as.vector(w*(Sigma %*% w))
  return(as.vector(r - theta))
}

A_rc_vs_theta <- function(wtheta, Sigma, b = NA, Sigma_w = NA) {
  N <- length(wtheta)-1
  w <- wtheta[1:N]
  Sigma_w <- as.vector(Sigma %*% w)
  Ut <- diag(Sigma_w) + Sigma * w
  return(cbind(Ut, -1))
}


################################################################
# Compute g, R, and A for the formulation "rc-over-b vs theta" #
################################################################

R_rc_over_b_vs_theta <- function(wtheta, Sigma, b, r = NA) {
  return(sum(g_rc_over_b_vs_theta(wtheta, Sigma, b)^2))
}

R_grad_rc_over_b_vs_theta <- function(wtheta, Sigma, b) {
  N <- length(wtheta)-1
  w <- wtheta[1:N]
  theta <- wtheta[N+1]
  Sigma_w <- as.vector(Sigma %*% w)
  r <- as.vector(w*Sigma_w)
  v <- 2*(r/b - theta)
  vb <- v/b
  return(c(as.vector(Sigma %*% (w*vb) + Sigma_w*vb), -sum(v)))
}

g_rc_over_b_vs_theta <- function(wtheta, Sigma, b, r = NA) {
  N <- length(wtheta)-1
  theta <- wtheta[N+1]
  w <- wtheta[1:N]
  r <- as.vector(w*(Sigma %*% w))
  return(as.vector(r/b - theta))
}

A_rc_over_b_vs_theta <- function(wtheta, Sigma, b, Sigma_w = NA) {
  N <- length(wtheta)-1
  w <- wtheta[1:N]
  Sigma_w <- as.vector(Sigma %*% w)
  Ut <- diag(Sigma_w) + Sigma * w
  return(cbind(Ut/b, -1))
}

Try the riskParityPortfolio package in your browser

Any scripts or data that you put into this service are public.

riskParityPortfolio documentation built on June 1, 2021, 9:07 a.m.