If any issues, questions or suggestions feel free to reach me out via e-mail or Linkedin. You can also visit my Github.

if(!require(pacman)) install.packages('pacman')
pacman::p_load(PolishStock, dplyr, ggplot2)

Assume there are \(N\) assets available with expected returns \(\mu = \left[ \mu_1, \dots, \mu_n \right]^T\) and covariance matrix \(\Sigma = \left[ \sigma_{i,j} \right]^N_{i,j=1}\). Portfolio is defined by weights vector \(\omega = \left[ \omega_1, \dots, \omega_n \right]^T\) such that \(\Sigma_{i=1}^N \omega_i = 1\). Then expected return of the portfolio is \[\mu_R = \omega^T\mu = \Sigma_{i=1}^N \omega_i \mu_i\] and variance of the portfolio is \[\sigma_R^2 = \omega^T \Sigma \omega = \Sigma_{i=1}^N \Sigma_{j=1}^N \omega_i \omega_k \sigma_{ij} \text{.}\] If \(N>2\) then set of attainable portfolios is 2-dimensional plane (assuming assets ain’t perfectly correlated) with convex left edge. Variance minimization problem is either quadratic programming problem (if short sale is forbidden) or using Lagrance multipliers can be reduced to the system of linear equations (if short sale is allowed). However, in this post we will simulate \(10000\) portfolios and select the optimal ones. Optimal means either portfolio with minimal variance or portfolio with maximal Sharpe ratio which is defined \[\text{Sharpe} = \frac{\mu_R - \mu_F}{\sigma_R} \text{,}\] where \(\mu_F\) is risk-free insterest rate. It can be easily verified that when putting \(\sigma_R\) formula into Sharpe ratio formula then lower correlation between asset returns gives us higher Sharpe ratio. In other words, portfolios with high sharpe ratio should be well-diversified.

We will build portfolios from 4 ETFs:
- iShares MSCI Emerging Markets ETF (EEM.US)
- iShares Gold Trust (IAU.US)
- iShares MSCI World ETF (URTH.US)
- iShares TIPS Bond ETF (TIP.US)

We use PolishStock package to download data from the website stooq.pl. Daily data for period January 2012 - February 2022 consists of 2345 observations.

tickers <- c('EEM.US', 'IAU.US', 'URTH.US', 'TIP.US')

for (ticker in tickers){
  stooq_download(ticker, '20120101', '20220228'
                 ,destination = paste0(getwd(), '/datasets/'))
}

close_prices <- df_prices(paste0(tickers, '.csv')
                          ,source = 'datasets/')

returns <- sapply(select(close_prices, -Date)
                  ,FUN = function(x) diff(log(x), lag = 1))

Although in the package PolishStock package there exists a function which does all the steps below (portfolio_simulations(close_prices, 1e4, risk_measure = ‘standard deviation’, risk_free = 0.02)), for educational purposes we write it explicitly below. First we calculate annualized average returns and covariance matrix of ETFs returns. Assume \(2\%\) risk-free interest rate.

mean_returns <- (colMeans(returns) + 1)^252 - 1
cov_returns <- cov(returns) * 252
sd_returns <- diag(cov_returns) %>% sqrt()
rf <- 0.02

We initialize matrix where portoflios weights will be stored and data frame where portfolios return, risk and sharpe ratio will be stored.

weights <- matrix(nrow = num_of_portfolios, ncol = ncol(returns))

portfolio_metrics <- data.frame(
  returns = rep(0, num_of_portfolios)
  ,risk = rep(0, num_of_portfolios)
  ,sharpe = rep(0, num_of_portfolios)
)

We simulate \(10000\) random portfolios with random weights from Uniform[0,1] distribution (which means that short sale is not allowed in our model). We calculate portfolio return, risk and sharpe ratio as explained above.

num_of_portfolios <- 1e4
set.seed(2137)
for (i in 1:num_of_portfolios) {
  
  random_weights <- runif(ncol(weights))
  random_weights <- random_weights / sum(random_weights)
  weights[i, ] <- random_weights
  
  portfolio_metrics$returns[i] <- random_weights %*% mean_returns
  portfolio_metrics$risk[i] <- sqrt(t(random_weights) %*% (cov_returns %*% random_weights))
  portfolio_metrics$sharpe[i] <- (portfolio_metrics$returns[i] - rf) / portfolio_metrics$risk[i]
  
}

Below we see optimal portfolios:
- minimal variance portfolio has only \(3.38\%\) expected annual return and mainly consists of treasury inflation-protected bonds
- maximal sharpe portfolio has \(7.23\%\) expected annual return and almost two times higher standard deviation than minimal variance portfolio. Miximal sharpe portfolio consists of developed markets stocks and treasury inflation-protected bonds in proportion 50:50.

optimal_portfolios <- data.frame(
  Min_Variance = c(
    round(100*weights[which.min(portfolio_metrics$risk),], 2)
    ,round(100*portfolio_metrics[which.min(portfolio_metrics$risk), ], 2) %>% as.numeric()
  )
  ,Max_Sharpe = c(
    round(100*weights[which.max(portfolio_metrics$sharpe),], 2)
    ,round(100*portfolio_metrics[which.max(portfolio_metrics$sharpe), ], 2) %>% as.numeric()
  )
) %>% t() %>% as.data.frame()

colnames(optimal_portfolios) <- c(colnames(returns), 'Expected Return', 'Standard Deviation', 'Sharpe Ratio')
optimal_portfolios

Below we see simulated portfolios on 2-dimensional risk-return plane. Minimal variance portfolio is green triangle and maximal sharpe portfolio is the orange one.

ggplot(portfolio_metrics, aes(x = risk, y = returns, color = sharpe)) +
  geom_point() +
  geom_point(aes(x = (optimal_portfolios$`Standard Deviation`[1] / 100)
                 ,y = (optimal_portfolios$`Expected Return`)[1] / 100)
             ,color = '#2ca02c', size = 5, shape = 17) +
  geom_point(aes(x = (optimal_portfolios$`Standard Deviation`[2] / 100)
                 ,y = (optimal_portfolios$`Expected Return`)[2] / 100)
             ,color = '#ff7f0e', size = 5, shape = 17) + 
  theme_bw()

---
title: "Portfolio Simulations"
output: html_notebook
---

If any issues, questions or suggestions feel free to reach me out via e-mail <wieczynskipawel@gmail.com> or [Linkedin](https://www.linkedin.com/in/pawel-wieczynski/). You can also visit my [Github](https://github.com/pawel-wieczynski).

```{r libraries, warning=FALSE, message=FALSE}
if(!require(pacman)) install.packages('pacman')
pacman::p_load(PolishStock, dplyr, ggplot2)
```

Assume there are $N$ assets available with expected returns $\mu = \left[ \mu_1, \dots, \mu_n \right]^T$ and covariance matrix $\Sigma = \left[ \sigma_{i,j} \right]^N_{i,j=1}$. Portfolio is defined by weights vector $\omega = \left[ \omega_1, \dots, \omega_n \right]^T$ such that $\Sigma_{i=1}^N \omega_i = 1$. Then expected return of the portfolio is $$\mu_R = \omega^T\mu = \Sigma_{i=1}^N \omega_i \mu_i$$ and variance of the portfolio is $$\sigma_R^2 = \omega^T \Sigma \omega = \Sigma_{i=1}^N \Sigma_{j=1}^N \omega_i \omega_k \sigma_{ij} \text{.}$$ If $N>2$ then set of attainable portfolios is 2-dimensional plane (assuming assets ain't perfectly correlated) with convex left edge. Variance minimization problem is either quadratic programming problem (if short sale is forbidden) or using Lagrance multipliers can be reduced to the system of linear equations (if short sale is allowed). However, in this post we will simulate $10000$ portfolios and select the optimal ones. *Optimal* means either portfolio with minimal variance or portfolio with maximal Sharpe ratio which is defined $$\text{Sharpe} = \frac{\mu_R - \mu_F}{\sigma_R} \text{,}$$ where $\mu_F$ is risk-free insterest rate. It can be easily verified that when putting $\sigma_R$ formula into Sharpe ratio formula then lower correlation between asset returns gives us higher Sharpe ratio. In other words, portfolios with high sharpe ratio should be well-diversified.

We will build portfolios from 4 ETFs: \
 - iShares MSCI Emerging Markets ETF (EEM.US) \
 - iShares Gold Trust (IAU.US) \
 - iShares MSCI World ETF (URTH.US) \
 - iShares TIPS Bond ETF (TIP.US)
 
We use [PolishStock package](https://github.com/pawel-wieczynski/PolishStock) to download data from the website [stooq.pl](https://stooq.pl/). Daily data for period January 2012 - February 2022 consists of 2345 observations.

```{r get_data, message=FALSE, warning=FALSE}
tickers <- c('EEM.US', 'IAU.US', 'URTH.US', 'TIP.US')

for (ticker in tickers){
  stooq_download(ticker, '20120101', '20220228'
                 ,destination = paste0(getwd(), '/datasets/'))
}

close_prices <- df_prices(paste0(tickers, '.csv')
                          ,source = 'datasets/')

returns <- sapply(select(close_prices, -Date)
                  ,FUN = function(x) diff(log(x), lag = 1))

```

Although in the package [PolishStock package](https://github.com/pawel-wieczynski/PolishStock) there exists a function which does all the steps below (*portfolio_simulations(close_prices, 1e4, risk_measure = 'standard deviation', risk_free = 0.02)*), for educational purposes we write it explicitly below. First we calculate annualized average returns and covariance matrix of ETFs returns. Assume $2\%$ risk-free interest rate.
```{r mean_and_covariance}
mean_returns <- (colMeans(returns) + 1)^252 - 1
cov_returns <- cov(returns) * 252
sd_returns <- diag(cov_returns) %>% sqrt()
rf <- 0.02
```

We initialize matrix where portoflios weights will be stored and data frame where portfolios return, risk and sharpe ratio will be stored.
```{r initialize_tables}
weights <- matrix(nrow = num_of_portfolios, ncol = ncol(returns))

portfolio_metrics <- data.frame(
  returns = rep(0, num_of_portfolios)
  ,risk = rep(0, num_of_portfolios)
  ,sharpe = rep(0, num_of_portfolios)
)
```

We simulate $10000$ random portfolios with random weights from Uniform[0,1] distribution (which means that short sale is not allowed in our model). We calculate portfolio return, risk and sharpe ratio as explained above.
```{r simulation}
num_of_portfolios <- 1e4
set.seed(2137)
for (i in 1:num_of_portfolios) {
  
  random_weights <- runif(ncol(weights))
  random_weights <- random_weights / sum(random_weights)
  weights[i, ] <- random_weights
  
  portfolio_metrics$returns[i] <- random_weights %*% mean_returns
  portfolio_metrics$risk[i] <- sqrt(t(random_weights) %*% (cov_returns %*% random_weights))
  portfolio_metrics$sharpe[i] <- (portfolio_metrics$returns[i] - rf) / portfolio_metrics$risk[i]
  
}
```

Below we see optimal portfolios: \
 - minimal variance portfolio has only $3.38\%$ expected annual return and mainly consists of treasury inflation-protected bonds \
 - maximal sharpe portfolio has $7.23\%$ expected annual return and almost two times higher standard deviation than minimal variance portfolio. Miximal sharpe portfolio consists of developed markets stocks and treasury inflation-protected bonds in proportion 50:50.
```{r optimal_portfolios}
optimal_portfolios <- data.frame(
  Min_Variance = c(
    round(100*weights[which.min(portfolio_metrics$risk),], 2)
    ,round(100*portfolio_metrics[which.min(portfolio_metrics$risk), ], 2) %>% as.numeric()
  )
  ,Max_Sharpe = c(
    round(100*weights[which.max(portfolio_metrics$sharpe),], 2)
    ,round(100*portfolio_metrics[which.max(portfolio_metrics$sharpe), ], 2) %>% as.numeric()
  )
) %>% t() %>% as.data.frame()

colnames(optimal_portfolios) <- c(colnames(returns), 'Expected Return', 'Standard Deviation', 'Sharpe Ratio')
optimal_portfolios
```

Below we see simulated portfolios on 2-dimensional risk-return plane. Minimal variance portfolio is green triangle and maximal sharpe portfolio is the orange one. 
```{r vizualization}
ggplot(portfolio_metrics, aes(x = risk, y = returns, color = sharpe)) +
  geom_point() +
  geom_point(aes(x = (optimal_portfolios$`Standard Deviation`[1] / 100)
                 ,y = (optimal_portfolios$`Expected Return`)[1] / 100)
             ,color = '#2ca02c', size = 5, shape = 17) +
  geom_point(aes(x = (optimal_portfolios$`Standard Deviation`[2] / 100)
                 ,y = (optimal_portfolios$`Expected Return`)[2] / 100)
             ,color = '#ff7f0e', size = 5, shape = 17) + 
  theme_bw()
```