Executive Summary
It is widely known the business of
portfolio management extends from separating the wheat from the chaff,
when it comes to identifying good investments, to the choice of finding
the optimal combinations of those identified investments. Arguably, what
makes a good portfolio manager is not how well they identify investment
opportunities but rather how they assign the investments relative
weights so that they make sense in a portfolio.
To answer the question of which weighting scheme provides the best
risk-adjusted return in a portfolio, a comparison of four(4) popular
weighting schemes is explored where it is found the Black-Litterman
Approach performs best for a hypothetical ETF portfolio among the Equal
Weight, Markowitz, Black-Litterman, and Covariance-Robust optimization
weighting schemes.
Assembling A Portfolio
After identifying attractive securities
for portfolio construction, it is typical for a portfolio manager to
consider how these securities might fit in a portfolio. This part of the
investment process often reduces to finding the “best” weights for the
various securities in the portfolio. The exact portfolio allocation
however, differs from one manager to the next with some preferring to
use ad hoc approaches for determining portfolio weights and others
relying on more structured risk-return analysis that often employs some
form of optimization modelling - a methodology for selecting an optimal
strategy given an objective and a set of constraints.
In terms of ad hoc approaches, the most widely used is the equal
weight approach where the amount invested in N candidate securities is
1/N of the available capital. For those who use optimization as a
weighting method, the general idea is to determine the weights of
securities in a portfolio that accomplish specific targets, usually in
the form of desired risk and return combinations, in the face of a set
of constraints and limitations faced by the portfolio manager. In both
of these approaches, however, it is generally assumed investors take a
single-period view of investing where the goal of portfolio allocation
is to invest optimally over some single predetermined period of
time.
As a case study for this article, the goal is to construct a
portfolio consisting of 9 ETFs: SPY(S&P 500), IWM(Russell 2000),
FEZ(Euro Stoxx 50), ACWI (MSCI All-Country World), AGG(Barclays
Aggregate US Bond), IAGG(iShares Core International Aggregate Bond),
IYR(Dow Jones US Real Estate), REET(iShares Global REIT), and
GSG(iShares GSCI Commodity Index); where it is explored how different
allocation procedures affect portfolio performance. The data used is
monthly price data gathered from the 2016-2021 period from Yahoo
Finance, which is evaluated on the 2022 price data. The weighting
schemes explored are
- Ad hoc:
- Optimization:
- Markowitz
- Black-Litterman
- Covariance-Robust
To make the analysis realistic, an
extended form of the above weighting schemes is explored where extended
is taken to mean accounting for possible constraints imposed on the
portfolio manager by either clients or regulators. The investing period
is assumed to be a year where it is assumed the investor never
rebalances their portfolio (i.e, does not change the allocation between
securities). While the assumption of no-rebalancing might seem
restricting, it is worth mentioning for such a short investment time
frame, taxes resulting from transactions could have a substantial impact
on portfolio performance.
The rest of the article, however, is organized as follows:
- Intuition Behind Optimization Methods
- Portfolio Constraints
- Portfolio Weights
- Overview of Portfolio Performance Metrics
- Performance Evaluation
- Risk Decomposition
Intuition Behind Optimization Methods
Markowitz(Mean-Variance Framework)
The classic Markowitz Mean-Variance
framework assumes investors make their allocation decisions based on
both the expected return and risk arising from their investments. Of
course, Markowitz defined risk as the variance of future returns which
for a portfolio was consistent of two parts - the variance of the
returns of individual assets as well as the covariances(equivalently,
the correlations) between those returns. In this framework, investing
all your money in assets that are strongly correlated is not a prudent
strategy even if the individual assets appear to be the equivalent of
“investment diamonds.” The reason is simple, if one asset performs worse
than expected, it is likely the other assets will also perform poorly
due to their high correlation.
Using this framework of future asset returns and variance from N
securities, the Markowitz Mean-Variance framework seeks to compare the
various portfolios that could be built from the N securities where the
portfolio with the lowest risk for a given level of expected return is
said to be optimal.
Of course, there are an infinite number of potential portfolios
that can be constructed from the various risk and return combinations.
Consequently, the problem is often reduced to using quadratic
programming to find the minimum-risk portfolio without explicitly
calculating every portfolio’s risk and return.
Black-Litterman Framework
Before covering the Black-Litterman
approach, it would be useful to note it is based on a Bayesian
estimation approach, i.e subjective interpretations of future
probabilities. For Bayesian approaches, a probability distribution known
as the prior is used to represent the investor’s knowledge about the
behavior of an asset’s returns before any data is observed. To
complement this view, observations of the asset’s returns behavior are
recorded to compute a new probability distribution known as the
‘posterior distribution’ of the asset’s returns future behavior.
In this context, the Black-Litterman approach, uses the same
framework of quadratic programming to find the minimum-risk portfolio
with the difference being the expected returns and variance are
generated from a combined view of observed data and the investor’s
subjective view of the behavior of future returns. Of course the
investor’s view can be expressed in either absolute terms - where the
investor has a view on individual asset performance or relative terms -
where the investor has a view on how an individual asset performs in
comparison to another. This ability to incorporate subjective views into
a portfolio expected return and variance is perhaps the most valuable
part of the framework.
In practice, the subjective views are often generated from a
Vector Autoregressive model of order p, a basic econometric model to
represent the returns time series of N assets.
Covariance-Robust Framework
An important omission in the explanation
of the Markowitz and Black Litterman Optimization frameworks is they do
not account for the anomalies that may exist in the input data, i.e the
portfolio returns and variance estimates. Of course, anomalies in this
case is a reference to those points that do not necessarily ‘stick’ out
in the fitted empirical mean, variance, and covariance estimates but
have substantial influence on those estimates from the returns data
set.
The importance of detecting and dealing with anomalies cannot be
underestimated because anomalies in the data can distort parameter
estimates, with the effect in the case of finding the optimal
risk-return combination for a portfolio, being the combinations more
appropriately describe the anomalies.
To see why this happens, an example would be useful. For example,
outliers, which are not picked up in the sample mean and covariance
estimates, pull the ordinary sample mean toward themselves, and
artificially inflate the ordinary sample covariance matrix. If we knew a
priori which points were anomalous, we could simply exclude them when
estimating the parameters. But, this information is often
unavailable.
This idea of minimizing the influence of anomalies in estimating
the mean and covariance matrix is what robust optimization seeks to
explore. The Minimum Covariance Determinant (MCD), a popular robust
optimization method, answers the question of estimating the right
empirical mean and covariance by taking all possible subsets of the
data, of a specified size, and estimates the mean and covariance matrix
for each subset. It then keeps the estimates for the subset whose
covariance matrix has the smallest determinant and scales them by a
‘consistency’ factor.
The underlying idea here is since the determinant of the
covariance matrix measures - roughly speaking - the spread of a returns
distribution, minimizing it is akin to selecting a subset of the data
that has the tightest distribution and consequently is less likely to be
affected by outliers.
Portfolio Constraints
In order to mimic a realistic portfolio
with realistic taken to mean constraints imposed by either clients or
regulators on a portfolio manager, additional constraints are made for
the optimization problem. In this article, the additional constraints
are
Long-Only Constraints - where the short selling is not allowed in
the portfolio.
Holding constraints - where a single ETF can be at most 40% of
the portfolio and at least 5% of the portfolio.
While imposing limits on the maximum
exposure on a single ETF is intuitive, it is worthwhile explaining the
lower bound. The underlying reason is to allow a holding size small yet
substantial enough that the position in the ETF can contribute to
portfolio performance. A more detailed reason however is that assets,
particularly stocks, are often traded in multiples of minimum
transaction costs or rounds and consequently, very small positions
cannot be realistically acquired.
# Library Imports - Package names
packages <- c("MASS", "xts", "quantmod", "PortfolioAnalytics",
"ROI", "ROI.plugin.glpk", "ROI.plugin.quadprog", "tidyverse",
"highcharter", "PerformanceAnalytics")
# Install packages if not yet installed - Neat trick from Antoine Soetewey's blog
installed_packages <- packages %in% rownames(installed.packages())
if (any(installed_packages == FALSE)) {
install.packages(packages[!installed_packages])
}
# Packages loading
suppressMessages(lapply(packages, library, character.only = TRUE))
#' import data for each symbol and convert to monthly prices
#' Note conversion to monthly automatically removes NA-VALUES
etf.symbols <- c("SPY", "IWM", "FEZ", "ACWI","AGG","IAGG", "IYR","REET","GSG")
adj.close <- 6 # 6th field is adjusted close
etf.prices <- getSymbols(etf.symbols[1], source="yahoo",
auto.assign=FALSE, return.class="xts")[,adj.close]
for (i in 2:length(etf.symbols)) {
etf.tmp <- getSymbols(etf.symbols[i], source="yahoo",
auto.assign=FALSE, return.class="xts")[,adj.close]
etf.prices <- cbind(etf.prices, etf.tmp)
}
colnames(etf.prices) <- etf.symbols
etf.prices<- suppressWarnings(to.monthly(etf.prices,
indexAt='lastof',
OHLC=FALSE))
# Calculate Returns and remove na value in first row
etf.rets.data <- Return.calculate(na.omit(etf.prices),method='log') %>%
na.omit()
# Keep Returns from 2016-2021 for in-sample estimation
etf.rets<- etf.rets.data["2016/2021"]
# Keep Returns from 2022 for out-of-sample testing
etf.rets.test<- etf.rets.data["2022"]
###### Portfolio Weighting
# Portfolio Object Specification - Set up portfolio with objective and constraints(Long Only)
#' GMV = Global Minimum Variance , EW = Equally Weight
#' RO = Robust Optimization , BL = Black Litterman
port.spec <- portfolio.spec(assets = colnames(etf.rets)) %>%
add.objective( type="risk", name="StdDev") %>%
add.objective( type="return", name="mean") %>%
add.constraint(type="full_investment") %>%
add.constraint( type="long_only") %>%
add.constraint(type = "box", min=0.05, max=0.4)
port.spec.EW <- portfolio.spec(assets = colnames(etf.rets)) %>%
add.constraint(type="full_investment") %>%
add.constraint( type="long_only") %>%
add.constraint(type = "box", min=0.05, max=0.4)
# Set Portfolio Moments for Optimization Functions
samp.moments<- function(R){
out<-set.portfolio.moments(R = R, portfolio=port.spec, method= 'sample')
return(out)
}
#' Views are generated from a VAR-2 model from the MTS package
#' R is an xts return object
bl.moments<- function(R){
data<- fortify.zoo(R)[,-1]
m1<- MTS::VAR(data,p=2,output = F) # From MTS package
pred<- MTS::VARpred(m1,1,output = F) # 1 period ahead forecast
pred<- pred$pred
n.assets<-ncol(R)
pick<-diag(n.assets)
out<- black.litterman(R= R,P = pick,Views = pred)
out$mu<- out$BLMu
out$sigma<- out$BLSigma
return(out)
}
#' Find Optimal Weights
#' Rf= 0
GMV.weights<- optimize.portfolio(R = etf.rets,portfolio= port.spec,maxSR=TRUE,
optimize_method = "ROI",trace = TRUE,momentFUN='samp.moments')
print("Optimal Markowitz weights")
print(sprintf("%s : %s", etf.symbols,
scales::percent(as.numeric(GMV.weights$weights))))
#' The following function is obtained from Ross Bennett's (2018) Paper
#' Estimates Covariance Matrix
sigma.robust<- function(R){
out<- list()
set.seed(1234)
out$sigma<- cov.rob(R,method='mcd',nsamp = 'best')$cov
return(out)
}
RO.weights <-optimize.portfolio(R = etf.rets,portfolio= port.spec,maxSR=TRUE,
optimize_method = "ROI",trace = TRUE,momentFUN='sigma.robust')
print("Covariance-Robust Optimization Weights")
print(sprintf("%s : %s", etf.symbols,
scales::percent(as.numeric(RO.weights$weights))))
EW.weights <- equal.weight(etf.rets, portfolio= port.spec.EW)
print("Equal weights")
print(sprintf("%s : %s", etf.symbols,
scales::percent(as.numeric(EW.weights$weights))))
#' The optimizer is not able to find a unique 'tangential' solution, so remove
#' Max Sharpe ratio objective.
#' This 'loosened' optimization implementation is then focused on maximizing only portfolio
#' return subject to constraints on weights.
BL.weights<-invisible(optimize.portfolio(R = etf.rets,portfolio= port.spec,
optimize_method = "ROI",trace = TRUE,momentFUN='bl.moments'))
print("Black-Litterman Weights")
print(sprintf("%s : %s", etf.symbols,
scales::percent(as.numeric(BL.weights$weights))))
Portfolio Weights
Applying the various weighting schemes, a
bar plot of portfolio weights is produced along with a table containing
the same information in detail. Broadly speaking, it can be seen except
for the Black-Litterman approach, the optimization approaches are
concentrated in the bond ETFs which one can argue heuristically is a
reasonable allocation given the market turbulence of 2020 and 2021. The
Black-Litterman approach however, appears to be concentrated in the
Commodity ETF.
# Create List of portfolio objects and display their weights in a barplot
port.list <- list(GMV.weights,EW.weights, RO.weights, BL.weights)
names(port.list) <- c("Markowitz","Equal-Weight", 'Covariance Robust Optimization',
'Black-Litterman')
chart.Weights(combine.optimizations(port.list), plot.type = "barplot",
main="Weights stacked by Asset (Long-only)")

# Put Weights in a data frame(Scale by 100 to get it in percentages)
weights<- cbind(GMV.weights$weights,EW.weights$weights,
RO.weights$weights, BL.weights$weights)
weights<- round(weights,4)*100
colnames(weights)<- names(port.list)
knitr::kable(weights,caption= "Table 1 - Weights in (%)")
Table 1 - Weights in (%)
| SPY |
22.11 |
11.11 |
13.36 |
5 |
| IWM |
5.00 |
11.11 |
10.92 |
25 |
| FEZ |
5.00 |
11.11 |
5.00 |
5 |
| ACWI |
5.00 |
11.11 |
5.00 |
5 |
| AGG |
40.00 |
11.11 |
10.73 |
5 |
| IAGG |
7.89 |
11.11 |
40.00 |
5 |
| IYR |
5.00 |
11.11 |
5.00 |
5 |
| REET |
5.00 |
11.11 |
5.00 |
5 |
| GSG |
5.00 |
11.11 |
5.00 |
40 |
Overview of Portfolio Performance Metrics
Of course, it would be pointless to
construct portfolios from the various weighting schemes without
assessing how each scheme affected the portfolio’s performance. To this
end, a brief summary of popular performance metrics is presented to
assess the value of each weighting scheme.
Annualized Return: Geometric average of annual
returns generated by a portfolio over a given investment period.
Active Return: Excess portfolio returns over some
specified benchmark. Note how it differs from excess return.
Excess Return: Excess portfolio returns over some
specified risk-free asset.
Annualized Standard Deviation: Geometric average of
variability of a portfolio’s annual returns over a given investment
period. The higher the standard deviation of returns, the riskier the
portfolio weighting scheme is.
Beta(CAPM): Reflects risk of a portfolio in relation
to some benchmark. Statistically, it is the regression coefficient
reflecting how much a portfolio’s returns change for each unit change in
the benchmark’s returns. Higher betas mean the portfolio’s returns are
more sensitive to the benchmark’s returns.
Sharpe Ratio: Portfolio excess returns per unit of
risk. Excess returns here is the excess portfolio returns over some
specified risk-free asset while risk is the portfolio standard
deviation. In general, higher Sharpe ratios mean the portfolio provides
greater returns over the risk free asset while offering the same or less
variability of returns. Higher Sharpe ratios are thus usually
better.
Treynor Ratio: Portfolio excess returns per unit of
beta/benchmark risk. Like the Sharpe, excess returns here is the excess
portfolio returns over some specified risk-free asset. In general,
higher Treynor ratios mean the portfolio provides greater returns over
the risk free asset while offering less exposure to the benchmark.
Higher Treynor ratios are thus usually better.
Modigliani \(M^2\):
Adjusts each portfolio’s returns to what it would have been had the
portfolio manager taken the same amount of risk as the benchmark. In
general, higher Modigliani \(M^2\)
measures mean the portfolio provides greater returns while taking the
same amount risk as the benchmark where risk is the benchmark’s standard
deviation. Higher Modigliani \(M^2\)
measures are thus usually better.
Information Ratio: Ratio of excess portfolio returns
over the benchmark to the standard of the excess portfolio returns. In
general, higher information ratios mean the portfolio provides greater
excess returns over the benchmark per unit of risk. Higher information
ratios are thus usually better.
Tracking Error: Standard Deviation of excess
portfolio returns over the benchmark. The greater the tracking error,
the more the portfolio returns deviate from the benchmark.
Diversification Ratio: Ratio of the weighted average
of volatility individual assets to the overall portfolio volatility.
Here, weights are the portfolio weights. For a well diversified
long-only portfolio, the ratio should be greater than 1.
Concentration Ratio:
Ratio of the weighted sum of squared
individual asset volatility to weighted sum of individual asset
volatility squared. While confusing, the idea for the numerator is to
square the individual weighted volatilities before you sum, where
weights are the portfolio weights.
For the denominator, you take the weighted sum of individual asset
volatility before you take the square. For a well diversified, long-only
portfolio, the ratio should be less than 1. Additionally, portfolio
allocations with higher concentration ratios are judged to be more
concentrated than their weighting counterparts. Put simply, a lower
concentration ratio is usually better.
Customizing the Benchmark:
Since most investments establish a
benchmark to evaluate a portfolio manager’s performance, a customized
index is created for this purpose. The customized index is created to
better reflect the investor’s objective which for this article is a
desired exposure to Index ETFs.
Of course, this investor is assumed to desire exposure to indices
that track stocks, bonds, commodities, and real estate. To make the
analysis tractable, a price-weighted index is constructed which averages
equally the value of each Index ETF in the portfolio.
Performance Evaluation
Table 2 shows the historical performance
of each portfolio weighting scheme. It can be seen - generally speaking
- the Black-Litterman approach performs best for the investment period.
In particular, it produced better active and excess returns, higher
sharpe, treynor and information ratios among others. It is worth noting
while the Black-Litterman portfolio was well diversified, i.e in terms
of its diversification ratio, it had the most concentrated portfolio in
comparison to the other ETF portfolios.(from its concentration
ratio)
To show what this performance means for the investor, a line graph
is presented showing the growth of $10,000 in each portfolio weighting
scheme.
weights <- cbind(GMV.weights$weights,EW.weights$weights,
RO.weights$weights,BL.weights$weights)
# rets = returns
rets<-c()
#' Calculate Returns on Out-of-Sample Data - With Wealth Progression
#' Combine returns of each portfolio in a single data frame
for (i in 1:ncol(weights)) {
rets.values <- Return.portfolio(etf.rets.test, weights = weights[,i],
wealth.index = TRUE)
rets<-cbind(rets,rets.values)
}
#' Combine Returns into a single data frame
amount<- 10000
rets.scale<- rets*amount
date <- as.Date("2021-12-31")
init.investment <- xts(matrix(amount, nrow=1,ncol=4), order.by=date)
rets.scale<- rbind(init.investment, rets.scale)
names(rets.scale) <- c("Markowitz","Equal-Weight","Covariance-Robust Optimization",
"Black Litterman")
#' Performance - Cumulative Returns Chart
cum.rets.chart<-highchart(type='stock')%>%
hc_title(text='Hypothetical Growth of $10,000 ') %>%
hc_add_series(round(rets.scale[,1],2),
name=colnames(rets.scale)[1])%>%
hc_add_series(round(rets.scale[,2],2),
name=colnames(rets.scale)[2])%>%
hc_add_series(round(rets.scale[,3],2),
name=colnames(rets.scale)[3])%>%
hc_add_series(round(rets.scale[,4],2),
name=colnames(rets.scale)[4])%>%
hc_navigator(enabled=FALSE)%>%
hc_scrollbar(enabled=FALSE) %>%
hc_add_theme(hc_theme_flat())%>%
hc_exporting(enabled=TRUE)%>%
hc_legend(enabled=TRUE) %>%
hc_plotOptions(series = list(animation = FALSE))
cum.rets.chart
key.measures<- function( Ra, port.weight, bm, scale = 12,Sigma = cov(etf.rets), Rf= 0){
ret<- Return.annualized(Ra,scale = scale) *100
act.ret<- ActiveReturn(Ra ,Rb= bm,scale = scale) *100
ex.ret<- sum(Return.excess(Ra))*100 # Rf= 0
#' Note ret = ex.ret since Rf=0
#' That is, total returns should be roughly equal to excess returns
#' Where total returns = annualized returns if the investment period < or = 1 year
st.dev<- StdDev.annualized(Ra,weights=port.weight,scale = scale) *100
beta<- CAPM.beta(Ra,bm)
sharpe<- SharpeRatio.annualized(Ra, Rf= Rf, scale = scale)
treynor<- TreynorRatio(Ra, Rb = bm,scale=scale)
i.ratio<- InformationRatio(Ra, Rb = bm,scale=scale)
m.square<- Modigliani(Ra, Rb = bm,scale=scale)
t.error<- TrackingError(Ra, Rb = bm,scale = scale)
dr <- FRAPO::dr(port.weight, Sigma ) # Function obtained from FRAPO package
cr <- FRAPO::cr(port.weight, Sigma) # Function obtained from FRAPO package
res<- round(rbind(ret,act.ret,ex.ret, st.dev,beta,sharpe, treynor,m.square,i.ratio, t.error, dr,cr),3)
rownames(res) <- c("Annualized Return(%)","Active Return(%)","Excess Return(%)",
'Annualized Standard Deviation(%)',"Beta",'Sharpe','Treynor',
"M^2 of Modigliani", "Information Ratio","Tracking Error",
"Diversification Ratio","Concentration Ratio")
return(res)
}
#' Note the portfolio by construction is invested in the overall market.
#' The Benchmark used here is a Price-Weighted index.
#' Price- Weighted indices by construction average prices equally.
#' comp.ind = composite index returns
#' The motivation for this multi-asset composite index is the
#' FTSE Multi-Asset Composite Index Series which
#' according to FTSE Russell
#' is designed to measure cross-asset market returns for a range of risk exposures.
etf.prices$comp<-rowMeans(etf.prices)
#' Calculate Returns and remove na value in first row
#' Select Returns from 2022
comp.ind <- Return.calculate(etf.prices$comp,method='log') %>%
na.omit()
comp.ind<- comp.ind['2022']
# rets = returns
rets<-c()
#' Calculate Returns on Out-of-Sample Data - Without Wealth Progression
#' Combine returns of each portfolio in a single data frame
for (i in 1:ncol(weights)) {
rets.values <- Return.portfolio(etf.rets.test, weights = weights[,i])
rets<-cbind(rets,rets.values)
}
# ports = portfolios
ports<-c()
# Combine performance measures of each portfolio in a single data frame
for (i in 1:ncol(weights)) {
measures <- key.measures(port.weight = weights[,i],
Ra = rets[,i], bm = comp.ind)
ports<-cbind(ports,measures)
}
#' Note for the benchmark, all we need are the annualized
#' Returns, StDev, and Excess Returns
#' Since the benchmark is price-weighted, the weights used in the function is the Equal Weight
bm <- key.measures(port.weight = EW.weights$weights,
Ra = comp.ind, bm = comp.ind)
keep.rows<- c("Annualized Return(%)","Excess Return(%)",
'Annualized Standard Deviation(%)')
# Assign NA to those values not in keep.rows
bm[!(bm %in% bm[keep.rows,]),]<- NA
all.ports <- cbind(ports,bm)
colnames(all.ports) <- append(names(rets.scale),"Benchmark")
knitr::kable(all.ports,
caption="Table 2 - Key Portfolio Performance Measures")
Table 2 - Key Portfolio Performance Measures
| Annualized Return(%) |
-12.817 |
-12.955 |
-12.087 |
-3.541 |
-16.231 |
| Active Return(%) |
3.414 |
3.275 |
4.143 |
12.690 |
NA |
| Excess Return(%) |
-12.776 |
-12.515 |
-12.046 |
-2.342 |
-15.766 |
| Annualized Standard Deviation(%) |
13.645 |
16.604 |
12.901 |
16.405 |
19.814 |
| Beta |
0.675 |
0.823 |
0.645 |
0.669 |
NA |
| Sharpe |
-0.939 |
-0.780 |
-0.937 |
-0.216 |
NA |
| Treynor |
-0.190 |
-0.157 |
-0.187 |
-0.053 |
NA |
| M^2 of Modigliani |
-0.015 |
-0.012 |
-0.015 |
-0.002 |
NA |
| Information Ratio |
0.489 |
0.699 |
0.570 |
1.087 |
NA |
| Tracking Error |
0.070 |
0.047 |
0.073 |
0.117 |
NA |
| Diversification Ratio |
1.282 |
1.189 |
1.282 |
1.145 |
NA |
| Concentration Ratio |
0.163 |
0.135 |
0.138 |
0.336 |
NA |
Risk Decomposition
As a conclusion to this article, it would
be useful to examine how exposures to individual ETFs affect the
variability portfolio returns for each weighting scheme. This breakdown
of portfolio risk into contributions by individual ETFs when combined
with the portfolio weights shows whether historically the portfolio has
been over or underexposed to specific ETFs.
Should our hypothetical investor desire to re-allocate their
capital among the ETFs for a given weighting scheme, they can use this
risk decomposition information to increase or decrease portfolio
allocation to specific ETFs. For example, while the SPY makes up 22% of
the Markowitz ETF portfolio, it accounts for 35% of the portfolio return
volatility. Our investor might consider reducing exposure to the SPY on
this basis.
Table 3 shows the risk decomposition of each portfolio weighting
scheme.
##### Component Contribution to Portfolio Volatility/ Standard Deviation
# What percentage of portfolio St.Dev did each position account for?
all.comps.sd<-c()
# Combine component contributions of each portfolio in a single data frame
for (i in 1:ncol(weights)) {
# Scale by 100 to get percent from decimals like say 0.2 to 20%
Std <- (StdDev(etf.rets.test, weights = weights[,i], portfolio_method=
"component")$pct_contrib_StdDev)*100
all.comps.sd<-cbind(all.comps.sd,Std)
}
colnames(all.comps.sd) <- names(rets.scale)
knitr::kable(all.comps.sd,
caption ="Table 3 - Component Contribution to Portfolio Volatility/ Standard Deviation(%)")
Table 3 - Component Contribution to Portfolio Volatility/
Standard Deviation(%)
| SPY |
34.394427 |
14.230269 |
22.081628 |
5.827865 |
| IWM |
7.918145 |
14.507161 |
18.708315 |
30.750616 |
| FEZ |
9.392703 |
16.798717 |
9.741107 |
6.280780 |
| ACWI |
7.506255 |
13.570762 |
7.877084 |
5.389567 |
| AGG |
20.240645 |
4.039012 |
5.381124 |
1.197094 |
| IAGG |
3.379680 |
3.653431 |
17.929786 |
1.244528 |
| IYR |
7.518222 |
13.891722 |
7.936467 |
5.348458 |
| REET |
7.618834 |
14.050253 |
8.049772 |
5.618056 |
| GSG |
2.031089 |
5.258672 |
2.294717 |
38.343036 |
Resources Used
Bacon, Carl R. Practical Risk-Adjusted Performance Measurement.
Second edition. Wiley Finance Series. Hoboken, NJ: Wiley, 2022.
Fabozzi, Frank J., and Dessislava A. Pachamanova. Portfolio
Construction and Analytics. Frank J. Fabozzi Series. Hoboken, New
Jersey: John Wiley & Sons, Inc, 2016.
Black F, Litterman R (1992) Global portfolio optimization. Financial
Analysts J. 48(5):28–43.
Rosenthal, Dale W. R. A Quantitative Primer on Investments with r.
Chicago, IL: Q36 LLC, 2018.
Harry Markowitz, 1952. “Portfolio Selection,” Journal of Finance,
American Finance Association, vol. 7(1), pages 77-91, March.
Bennett Ross, 2018. “Custom Moment and Objective Functions”
Ali Pardhan, “Intuitive Explanation of Minimum Covariance Determinant
(MCD).” Cross Validated, July 5, 2020. https://stats.stackexchange.com/questions/475636/intuitive-explanation-of-minimum-covariance-determinant-mcd.
Brian G. Peterson and Peter Carl (2018). PortfolioAnalytics:
Portfolio Analysis, Including Numerical Methods for Optimization of
Portfolios. R package version 1.1.0. https://CRAN.R-project.org/package=PortfolioAnalytics
Brian G. Peterson and Peter Carl (2020). PerformanceAnalytics:
Econometric Tools for Performance and Risk Analysis. R package version
2.0.4. https://CRAN.R-project.org/package=PerformanceAnalytics
---
title: "Portfolio Construction and Analytics"
author: Kobena Amoah
date: "`r format(Sys.time(), '%d %B, %Y')`"
output: 
  html_document:
      toc: true
      toc_depth: 2
      code_folding: hide
      code_download: true
      keep_md: true 
always_allow_html: yes
---
## Executive Summary

|       It is widely known the business of portfolio management extends from separating the wheat from the chaff, when it comes to identifying good investments, to the choice of finding the optimal combinations of those identified investments. Arguably, what makes a good portfolio manager is not how well they identify investment opportunities but rather how they assign the investments relative weights so that they make sense in a portfolio.
|       To answer the question of which weighting scheme provides the best risk-adjusted return in a portfolio, a comparison of four(4) popular weighting schemes is explored where it is found the Black-Litterman Approach performs best for a hypothetical ETF portfolio among the Equal Weight, Markowitz, Black-Litterman, and Covariance-Robust optimization weighting schemes.

## Assembling A Portfolio
|       After identifying attractive securities for portfolio construction, it is typical for a portfolio manager to consider how these securities might fit in a portfolio. This part of the investment process often reduces to finding the "best" weights for the various securities in the portfolio. The exact portfolio allocation however, differs from one manager to the next with some preferring to use ad hoc approaches for determining portfolio weights and others relying on more structured risk-return analysis that often employs some form of optimization modelling - a methodology for selecting an optimal strategy given an objective and a set of constraints. 
|       In terms of ad hoc approaches, the most widely used is the equal weight approach where the amount invested in N candidate securities is 1/N of the available capital. For those who use optimization as a weighting method, the general idea is to determine the weights of securities in a portfolio that accomplish specific targets, usually in the form of desired risk and return combinations, in the face of a set of constraints and limitations faced by the portfolio manager. In both of these approaches, however, it is generally assumed investors take a single-period view of investing where the goal of portfolio allocation is to invest optimally over some single predetermined period of time. 
|       As a case study for this article, the goal is to construct a portfolio consisting of 9 ETFs: SPY(S&P 500), IWM(Russell 2000), FEZ(Euro Stoxx 50), ACWI (MSCI All-Country World), AGG(Barclays Aggregate US Bond), IAGG(iShares Core  International Aggregate Bond), IYR(Dow Jones US Real Estate), REET(iShares Global REIT), and GSG(iShares GSCI Commodity Index); where it is explored how different allocation procedures affect portfolio performance. The data used is monthly price data gathered from the 2016-2021 period from Yahoo Finance, which is evaluated on the 2022 price data. The weighting schemes explored are

a) Ad hoc:
   - Equal Weight
   
b) Optimization: 
   - Markowitz
   - Black-Litterman
   - Covariance-Robust
   
|       To make the analysis realistic, an extended form of the above weighting schemes is explored where extended is taken to mean accounting for possible constraints imposed on the portfolio manager by either clients or regulators. The investing period is assumed to be a year where it is assumed the investor never rebalances their portfolio (i.e, does not change the allocation between securities). While the assumption of no-rebalancing might seem restricting, it is worth mentioning for such a short investment time frame, taxes resulting from transactions could have a substantial impact on portfolio performance. 
|       The rest of the article, however, is organized as follows:
- Intuition Behind Optimization Methods
- Portfolio Constraints
- Portfolio Weights
- Overview of Portfolio Performance Metrics
- Performance Evaluation
- Risk Decomposition
 
## Intuition Behind Optimization Methods

**Markowitz(Mean-Variance Framework)**

|       The classic Markowitz Mean-Variance framework assumes investors make their allocation decisions based on both the expected return and risk arising from their investments. Of course, Markowitz defined risk as the variance of future returns which for a portfolio was consistent of two parts - the variance of the returns of individual assets as well as the covariances(equivalently, the correlations) between those returns. In this framework, investing all your money in assets that are strongly correlated is not a prudent strategy even if the individual assets appear to be the equivalent of "investment diamonds." The reason is simple, if one asset performs worse than expected, it is likely the other assets will also perform poorly due to their high correlation.
|       Using this framework of future asset returns and variance from N securities, the  Markowitz Mean-Variance framework seeks to compare the various portfolios that could be built from the N securities where the portfolio with the lowest risk for a given level of expected return is said to be optimal.
|       Of course, there are an infinite number of potential portfolios that can be constructed from the various risk and return combinations. Consequently, the problem is often reduced to using quadratic programming to find the minimum-risk portfolio without explicitly calculating every portfolio's risk and return.
  
**Black-Litterman Framework**

|       Before covering the Black-Litterman approach, it would be useful to note it is based on a Bayesian estimation approach, i.e subjective interpretations of future probabilities. For Bayesian approaches, a probability distribution known as the prior is used to represent the investor's knowledge about the behavior of an asset's returns before any data is observed. To complement this view, observations of the asset's returns behavior are recorded to compute a new probability distribution known as the 'posterior distribution' of the asset's returns future behavior.
|       In this context, the Black-Litterman approach, uses the same framework of quadratic programming to find the minimum-risk portfolio with the difference being the expected returns and variance are generated from a combined view of observed data and the investor's subjective view of the behavior of future returns. Of course the investor's view can be expressed in either absolute terms - where the investor has a view on individual asset performance or relative terms - where the investor has a view on how an individual asset performs in comparison to another. This ability to incorporate subjective views into a portfolio expected return and variance is perhaps the most valuable part of the framework.
|       In practice, the subjective views are often generated from a Vector Autoregressive model of order p, a basic econometric model to represent the returns time series of N assets.

**Covariance-Robust Framework**

|       An important omission in the explanation of the Markowitz and Black Litterman Optimization frameworks is they do not account for the anomalies that may exist in the input data, i.e the portfolio returns and variance estimates. Of course, anomalies in this case is a reference to those points that do not necessarily 'stick' out in the fitted empirical mean, variance, and covariance estimates but have substantial influence on those estimates from the returns data set.
|       The importance of detecting and dealing with anomalies cannot be underestimated because anomalies in the data can distort parameter estimates, with the effect in the case of finding the optimal risk-return combination for a portfolio, being the combinations more appropriately describe the anomalies.
|       To see why this happens, an example would be useful. For example, outliers, which are not picked up in the sample mean and covariance estimates, pull the ordinary sample mean toward themselves, and artificially inflate the ordinary sample covariance matrix. If we knew a priori which points were anomalous, we could simply exclude them when estimating the parameters. But, this information is often unavailable.
|       This idea of minimizing the influence of anomalies in estimating the mean and covariance matrix is what robust optimization seeks to explore. The Minimum Covariance Determinant (MCD), a popular robust optimization method, answers the question of estimating the right empirical mean and covariance by taking all possible subsets of the data, of a specified size, and estimates the mean and covariance matrix for each subset. It then keeps the estimates for the subset whose covariance matrix has the smallest determinant and scales them by a 'consistency' factor.
|       The underlying idea here is since the determinant of the covariance matrix measures - roughly speaking - the spread of a returns distribution, minimizing it is akin to selecting a subset of the data that has the tightest distribution and consequently is less likely to be affected by outliers.

## Portfolio Constraints

|       In order to mimic a realistic portfolio with realistic taken to mean constraints imposed by either clients or regulators on a portfolio manager, additional constraints are made for the optimization problem. In this article, the additional constraints are

a) Long-Only Constraints - where the short selling is not allowed in the portfolio.

b) Holding constraints - where a single ETF can be at most 40% of the portfolio and at least 5% of the portfolio.

|       While imposing limits on the maximum exposure on a single ETF is intuitive, it is worthwhile explaining the lower bound. The underlying reason is to allow a holding size small yet substantial enough that the position in the ETF can contribute to portfolio performance. A more detailed reason however is that assets, particularly stocks, are often traded in multiples of minimum transaction costs or rounds and consequently, very small positions cannot be realistically acquired.

```{r setup,warning=FALSE,comment=NA,results='hide'}
# Library Imports - Package names
packages <- c("MASS", "xts", "quantmod", "PortfolioAnalytics",
              "ROI", "ROI.plugin.glpk", "ROI.plugin.quadprog", "tidyverse", 
              "highcharter", "PerformanceAnalytics")

# Install packages if not yet installed - Neat trick from Antoine Soetewey's blog
installed_packages <- packages %in% rownames(installed.packages())
if (any(installed_packages == FALSE)) {
  install.packages(packages[!installed_packages])
}

# Packages loading
suppressMessages(lapply(packages, library, character.only = TRUE))

#' import data for each symbol and convert to monthly prices
#' Note conversion to monthly automatically removes NA-VALUES
etf.symbols <- c("SPY", "IWM", "FEZ", "ACWI","AGG","IAGG", "IYR","REET","GSG")
adj.close <- 6  # 6th field is adjusted close
etf.prices <- getSymbols(etf.symbols[1], source="yahoo",
                         auto.assign=FALSE, return.class="xts")[,adj.close]
for (i in 2:length(etf.symbols)) {
  etf.tmp <- getSymbols(etf.symbols[i], source="yahoo",
                        auto.assign=FALSE, return.class="xts")[,adj.close]
  etf.prices <- cbind(etf.prices, etf.tmp)
}
colnames(etf.prices) <- etf.symbols

etf.prices<- suppressWarnings(to.monthly(etf.prices,
                        indexAt='lastof',
                        OHLC=FALSE))

# Calculate Returns and remove na value in first row
etf.rets.data <- Return.calculate(na.omit(etf.prices),method='log') %>%
                na.omit()

# Keep Returns from 2016-2021 for in-sample estimation
etf.rets<- etf.rets.data["2016/2021"]

# Keep Returns from 2022 for out-of-sample testing
etf.rets.test<- etf.rets.data["2022"]

###### Portfolio Weighting

#  Portfolio Object Specification - Set up portfolio with objective and constraints(Long Only)

#'  GMV = Global Minimum Variance  ,    EW  = Equally Weight
#'  RO  = Robust Optimization ,   BL  = Black Litterman

port.spec <- portfolio.spec(assets = colnames(etf.rets)) %>%
                add.objective( type="risk", name="StdDev") %>%
                add.objective( type="return", name="mean") %>%
                add.constraint(type="full_investment")  %>%
                add.constraint( type="long_only")  %>%
                add.constraint(type = "box", min=0.05, max=0.4)

port.spec.EW <- portfolio.spec(assets = colnames(etf.rets)) %>%
                  add.constraint(type="full_investment")  %>%
                  add.constraint( type="long_only")  %>%
                  add.constraint(type = "box", min=0.05, max=0.4)

# Set Portfolio Moments for Optimization Functions
samp.moments<-  function(R){
    out<-set.portfolio.moments(R = R, portfolio=port.spec, method= 'sample')
    return(out)
}


#' Views are generated from a VAR-2 model from the MTS package
#' R is an xts return object
bl.moments<- function(R){
  data<- fortify.zoo(R)[,-1]
  m1<- MTS::VAR(data,p=2,output = F) # From MTS package
  pred<- MTS::VARpred(m1,1,output = F) # 1 period ahead forecast
  pred<- pred$pred   
  n.assets<-ncol(R)
  pick<-diag(n.assets)
  out<- black.litterman(R= R,P = pick,Views = pred)
  out$mu<- out$BLMu
  out$sigma<- out$BLSigma
  return(out)
}

#' Find Optimal Weights
#' Rf= 0 
GMV.weights<- optimize.portfolio(R = etf.rets,portfolio= port.spec,maxSR=TRUE,
                         optimize_method = "ROI",trace = TRUE,momentFUN='samp.moments')
 print("Optimal Markowitz weights")
 print(sprintf("%s : %s", etf.symbols,
              scales::percent(as.numeric(GMV.weights$weights))))

#' The following function is obtained from Ross Bennett's (2018) Paper
#' Estimates Covariance Matrix
sigma.robust<- function(R){
  out<- list()
  set.seed(1234)
  out$sigma<- cov.rob(R,method='mcd',nsamp = 'best')$cov
  return(out)
}

RO.weights <-optimize.portfolio(R = etf.rets,portfolio= port.spec,maxSR=TRUE,
                                 optimize_method = "ROI",trace = TRUE,momentFUN='sigma.robust')

 print("Covariance-Robust Optimization Weights")
 print(sprintf("%s : %s", etf.symbols,
              scales::percent(as.numeric(RO.weights$weights))))


EW.weights <- equal.weight(etf.rets, portfolio= port.spec.EW)
 print("Equal weights")
 print(sprintf("%s : %s", etf.symbols,
              scales::percent(as.numeric(EW.weights$weights))))
 
#' The optimizer is not able to find a unique 'tangential' solution, so remove 
#' Max Sharpe ratio objective.
#' This 'loosened' optimization implementation is then focused on maximizing only portfolio
#' return subject to constraints on weights.
BL.weights<-invisible(optimize.portfolio(R = etf.rets,portfolio= port.spec,
                               optimize_method = "ROI",trace = TRUE,momentFUN='bl.moments'))
 print("Black-Litterman Weights")
 print(sprintf("%s : %s", etf.symbols,
              scales::percent(as.numeric(BL.weights$weights))))
```
 
## Portfolio Weights

|       Applying the various weighting schemes, a bar plot of portfolio weights is produced along with a table containing the same information in detail. Broadly speaking, it can be seen except for the Black-Litterman approach, the optimization approaches are concentrated in the bond ETFs which one can argue heuristically is a reasonable allocation given the market turbulence of 2020 and 2021. The Black-Litterman approach however, appears to be concentrated in the Commodity ETF.
```{r 2nd,comment=NA, fig.dim=c(8,6),warning=FALSE}
# Create List of portfolio objects and display their weights in a barplot
port.list <- list(GMV.weights,EW.weights, RO.weights, BL.weights)
names(port.list) <- c("Markowitz","Equal-Weight", 'Covariance Robust Optimization', 
                      'Black-Litterman')
chart.Weights(combine.optimizations(port.list), plot.type = "barplot", 
              main="Weights stacked by Asset (Long-only)")

# Put Weights in a data frame(Scale by 100 to get it in percentages)
weights<- cbind(GMV.weights$weights,EW.weights$weights,
                RO.weights$weights, BL.weights$weights)
weights<- round(weights,4)*100
colnames(weights)<- names(port.list)

knitr::kable(weights,caption= "Table 1 - Weights in (%)")
```

## Overview of Portfolio Performance Metrics

|       Of course, it would be pointless to construct portfolios from the various weighting schemes without assessing how each scheme affected the portfolio's performance. To this end, a brief summary of popular performance metrics is presented to assess the value of each weighting scheme.

**Annualized Return:** Geometric average of annual returns generated by a portfolio over a given investment period.

**Active Return:** Excess portfolio returns over some specified benchmark. Note how it differs from excess return.

**Excess Return:** Excess portfolio returns over some specified risk-free asset. 

**Annualized Standard Deviation:** Geometric average of variability of a portfolio's annual returns over a given investment period. The higher the standard deviation of returns, the riskier the portfolio weighting scheme is.

**Beta(CAPM):** Reflects risk of a portfolio in relation to some benchmark. Statistically, it is the regression coefficient reflecting how much a portfolio's returns change for each unit change in the benchmark's returns. Higher betas mean the portfolio's returns are more sensitive to the benchmark's returns.

**Sharpe Ratio:** Portfolio excess returns per unit of risk. Excess returns here is the excess portfolio returns over some specified risk-free asset while risk is the portfolio standard deviation. In general, higher Sharpe ratios mean the portfolio provides greater returns over the risk free asset while offering the same or less variability of returns. Higher Sharpe ratios are thus usually better.

**Treynor Ratio:** Portfolio excess returns per unit of beta/benchmark risk. Like the Sharpe, excess returns here is the excess portfolio returns over some specified risk-free asset. In general, higher Treynor ratios mean the portfolio provides greater returns over the risk free asset while offering less exposure to the benchmark. Higher Treynor ratios are thus usually better.

**Modigliani** $M^2$:  Adjusts each portfolio's returns to what it would have been had the portfolio manager taken the same amount of risk as the benchmark. In general, higher Modigliani $M^2$ measures mean the portfolio provides greater returns while taking the same amount risk as the benchmark where risk is the benchmark's standard deviation. Higher Modigliani $M^2$ measures are thus usually better.

**Information Ratio:** Ratio of excess portfolio returns over the benchmark to the standard of the excess portfolio returns. In general, higher information ratios mean the portfolio provides greater excess returns over the benchmark per unit of risk. Higher information ratios are thus usually better.

**Tracking Error:** Standard Deviation of excess portfolio returns over the benchmark. The greater the tracking error, the more the portfolio returns deviate from the benchmark.

**Diversification Ratio:** Ratio of the weighted average of volatility individual assets to the overall portfolio volatility. Here, weights are the portfolio weights. For a well diversified long-only portfolio, the ratio should be greater than 1.

**Concentration Ratio:** 

|       Ratio of the weighted sum of squared individual asset volatility to weighted sum of individual asset volatility squared. While confusing, the idea for the numerator is to square the individual weighted volatilities before you sum, where weights are the portfolio weights.
|       For the denominator, you take the weighted sum of individual asset volatility before you take the square. For a well diversified, long-only portfolio, the ratio should be less than 1. Additionally, portfolio allocations with higher concentration ratios are judged to be more concentrated than their weighting counterparts. Put simply, a lower concentration ratio is usually better.

**Customizing the Benchmark:**

|       Since most investments establish a benchmark to evaluate a portfolio manager's performance, a customized index is created for this purpose. The customized index is created to better reflect the investor's objective which for this article is a desired exposure to Index ETFs.
|       Of course, this investor is assumed to desire exposure to indices that track stocks, bonds, commodities, and real estate. To make the analysis tractable, a price-weighted index is constructed which averages equally the value of each Index ETF in the portfolio.

## Performance Evaluation

|       Table 2 shows the historical performance of each portfolio weighting scheme. It can be seen - generally speaking - the Black-Litterman approach performs best for the investment period. In particular, it produced better active and excess returns, higher sharpe, treynor and information ratios among others. It is worth noting while the Black-Litterman portfolio was well diversified, i.e in terms of its diversification ratio, it had the most concentrated portfolio in comparison to the other ETF portfolios.(from its concentration ratio) 
|       To show what this performance means for the investor, a line graph is presented showing the growth of $10,000 in each portfolio weighting scheme.
```{r 3rd,fig.cap="Cumulated Portoflio Returns",comment=NA,fig.dim=c(8,6),warning=FALSE,dev='png'}
weights <- cbind(GMV.weights$weights,EW.weights$weights,
                 RO.weights$weights,BL.weights$weights)
# rets = returns
rets<-c()

#' Calculate Returns on Out-of-Sample Data - With Wealth Progression
#' Combine returns of each portfolio in a single data frame
for (i in 1:ncol(weights)) {
  rets.values <- Return.portfolio(etf.rets.test, weights  = weights[,i],
                                  wealth.index = TRUE)
  rets<-cbind(rets,rets.values)
}

#' Combine Returns into a single data frame
amount<- 10000
rets.scale<- rets*amount
date <- as.Date("2021-12-31")
init.investment <- xts(matrix(amount, nrow=1,ncol=4), order.by=date)
rets.scale<- rbind(init.investment, rets.scale)
names(rets.scale) <- c("Markowitz","Equal-Weight","Covariance-Robust Optimization",
                          "Black Litterman")

#' Performance - Cumulative Returns Chart
cum.rets.chart<-highchart(type='stock')%>%
  hc_title(text='Hypothetical Growth of $10,000 ') %>%
  hc_add_series(round(rets.scale[,1],2), 
                name=colnames(rets.scale)[1])%>%
  hc_add_series(round(rets.scale[,2],2), 
                name=colnames(rets.scale)[2])%>%
  hc_add_series(round(rets.scale[,3],2), 
                name=colnames(rets.scale)[3])%>%
  hc_add_series(round(rets.scale[,4],2), 
                name=colnames(rets.scale)[4])%>%
  hc_navigator(enabled=FALSE)%>%
  hc_scrollbar(enabled=FALSE) %>%
  hc_add_theme(hc_theme_flat())%>%
  hc_exporting(enabled=TRUE)%>%
  hc_legend(enabled=TRUE) %>% 
  hc_plotOptions(series = list(animation = FALSE))
cum.rets.chart


key.measures<- function( Ra, port.weight, bm, scale = 12,Sigma = cov(etf.rets), Rf= 0){
  ret<-     Return.annualized(Ra,scale = scale) *100
  act.ret<-  ActiveReturn(Ra ,Rb= bm,scale = scale) *100
  ex.ret<-   sum(Return.excess(Ra))*100   # Rf= 0 
  #' Note ret = ex.ret since Rf=0
  #' That is, total returns should be roughly equal to excess returns
  #' Where total returns = annualized returns if the investment period < or = 1 year
  
  st.dev<-  StdDev.annualized(Ra,weights=port.weight,scale = scale) *100
  beta<- CAPM.beta(Ra,bm)
  sharpe<-  SharpeRatio.annualized(Ra, Rf= Rf, scale = scale)
  treynor<- TreynorRatio(Ra, Rb = bm,scale=scale)
  i.ratio<- InformationRatio(Ra, Rb = bm,scale=scale)
  m.square<- Modigliani(Ra, Rb = bm,scale=scale)
  t.error<- TrackingError(Ra, Rb = bm,scale = scale)
  dr <-     FRAPO::dr(port.weight, Sigma ) # Function obtained from FRAPO package
  cr <-     FRAPO::cr(port.weight, Sigma)  # Function obtained from FRAPO package
  res<-     round(rbind(ret,act.ret,ex.ret, st.dev,beta,sharpe, treynor,m.square,i.ratio, t.error, dr,cr),3)

  rownames(res) <- c("Annualized Return(%)","Active Return(%)","Excess Return(%)",
                     'Annualized Standard Deviation(%)',"Beta",'Sharpe','Treynor',
                  "M^2 of Modigliani", "Information Ratio","Tracking Error",
                  "Diversification Ratio","Concentration Ratio")
  return(res)
}

#'   Note the portfolio by construction is invested in the overall market.
#'   The Benchmark used here is a Price-Weighted index.
#'   Price- Weighted indices by construction average prices equally.
#'   comp.ind = composite index returns

#'   The motivation for this multi-asset composite index is the
#'   FTSE Multi-Asset Composite Index Series which 
#'   according to FTSE Russell
#'   is designed to measure cross-asset market returns for a range of risk exposures.

etf.prices$comp<-rowMeans(etf.prices)
#' Calculate Returns and remove na value in first row
#' Select Returns from 2022 
comp.ind <- Return.calculate(etf.prices$comp,method='log') %>%
            na.omit()
comp.ind<- comp.ind['2022']

# rets = returns
rets<-c()

#' Calculate Returns on Out-of-Sample Data - Without Wealth Progression
#' Combine returns of each portfolio in a single data frame
for (i in 1:ncol(weights)) {
  rets.values <- Return.portfolio(etf.rets.test, weights  = weights[,i])
  rets<-cbind(rets,rets.values)
}

# ports = portfolios
ports<-c()
# Combine performance measures of each portfolio in a single data frame
for (i in 1:ncol(weights)) {
  measures <- key.measures(port.weight = weights[,i], 
                 Ra = rets[,i], bm =  comp.ind)
  ports<-cbind(ports,measures)
}

#'  Note for the benchmark, all we need are the annualized
#'  Returns, StDev, and Excess Returns
#'  Since the benchmark is price-weighted, the weights used in the function is the Equal Weight
bm <- key.measures(port.weight = EW.weights$weights, 
                  Ra = comp.ind, bm =  comp.ind)

keep.rows<- c("Annualized Return(%)","Excess Return(%)",
         'Annualized Standard Deviation(%)')
# Assign NA to those values not in keep.rows
bm[!(bm %in% bm[keep.rows,]),]<- NA  
all.ports <- cbind(ports,bm)


colnames(all.ports) <- append(names(rets.scale),"Benchmark")
knitr::kable(all.ports,
             caption="Table 2 - Key Portfolio Performance Measures")
```


## Risk Decomposition
|       As a conclusion to this article, it would be useful to examine how exposures to individual ETFs affect the variability portfolio returns for each weighting scheme. This breakdown of portfolio risk into contributions by individual ETFs when combined with the portfolio weights shows whether historically the portfolio has been over or underexposed to specific ETFs. 
|       Should our hypothetical investor desire to re-allocate their capital among the ETFs for a given weighting scheme, they can use this risk decomposition information to increase or decrease portfolio allocation to specific ETFs. For example, while the SPY makes up 22% of the Markowitz ETF portfolio, it accounts for 35% of the portfolio return volatility. Our investor might consider reducing exposure to the SPY on this basis.
|       Table 3 shows the risk decomposition of each portfolio weighting scheme.
```{r 4th, comment=NA,fig.dim=c(8,6),warning=FALSE}
##### Component Contribution to Portfolio Volatility/ Standard Deviation

#   What percentage of portfolio St.Dev did each position account for?

all.comps.sd<-c()
# Combine component contributions of each portfolio in a single data frame
for (i in 1:ncol(weights)) {
  # Scale by 100 to get percent from decimals like say 0.2 to 20%
  Std <- (StdDev(etf.rets.test, weights = weights[,i], portfolio_method=  
                  "component")$pct_contrib_StdDev)*100
  all.comps.sd<-cbind(all.comps.sd,Std)
}

colnames(all.comps.sd) <- names(rets.scale)
knitr::kable(all.comps.sd,
             caption ="Table 3 - Component Contribution to Portfolio Volatility/ Standard Deviation(%)")
```

## Resources Used

Bacon, Carl R. Practical Risk-Adjusted Performance Measurement. Second edition. Wiley Finance Series. Hoboken, NJ: Wiley, 2022.

Fabozzi, Frank J., and Dessislava A. Pachamanova. Portfolio Construction and Analytics. Frank J. Fabozzi Series. Hoboken, New Jersey: John Wiley & Sons, Inc, 2016.

Black F, Litterman R (1992) Global portfolio optimization. Financial Analysts J. 48(5):28–43.

Rosenthal, Dale W. R. A Quantitative Primer on Investments with r. Chicago, IL: Q36 LLC, 2018.

Harry Markowitz, 1952. "Portfolio Selection," Journal of Finance, American Finance Association, vol. 7(1), pages 77-91, March.

Bennett Ross, 2018. "Custom Moment and Objective Functions"

Ali Pardhan, “Intuitive Explanation of Minimum Covariance Determinant (MCD).” Cross Validated, July 5, 2020. https://stats.stackexchange.com/questions/475636/intuitive-explanation-of-minimum-covariance-determinant-mcd. 

Brian G. Peterson and Peter Carl (2018). PortfolioAnalytics: Portfolio Analysis,
  Including Numerical Methods for Optimization of Portfolios. R package version 1.1.0.
  https://CRAN.R-project.org/package=PortfolioAnalytics

Brian G. Peterson and Peter Carl (2020). PerformanceAnalytics: Econometric Tools for
  Performance and Risk Analysis. R package version 2.0.4.
  https://CRAN.R-project.org/package=PerformanceAnalytics