Source code for pypfopt.objective_functions

"""
The ``objective_functions`` module provides optimization objectives, including the actual
objective functions called by the ``EfficientFrontier`` object's optimization methods.
These methods are primarily designed for internal use during optimization and each requires
a different signature (which is why they have not been factored into a class).
For obvious reasons, any objective function must accept ``weights``
as an argument, and must also have at least one of ``expected_returns`` or ``cov_matrix``.

The objective functions either compute the objective given a numpy array of weights, or they
return a cvxpy *expression* when weights are a ``cp.Variable``. In this way, the same objective
function can be used both internally for optimization and externally for computing the objective
given weights. ``_objective_value()`` automatically chooses between the two behaviours.

``objective_functions`` defaults to objectives for minimisation. In the cases of objectives
that clearly should be maximised (e.g Sharpe Ratio, portfolio return), the objective function
actually returns the negative quantity, since minimising the negative is equivalent to maximising
the positive. This behaviour is controlled by the ``negative=True`` optional argument.

Currently implemented:

- Portfolio variance (i.e square of volatility)
- Portfolio return
- Sharpe ratio
- L2 regularisation (minimising this reduces nonzero weights)
- Quadratic utility
- Transaction cost model (a simple one)
- Ex-ante (squared) tracking error
- Ex-post (squared) tracking error
"""

import numpy as np
import cvxpy as cp


def _objective_value(w, obj):
    """
    Helper method to return either the value of the objective function
    or the objective function as a cvxpy object depending on whether
    w is a cvxpy variable or np array.

    :param w: weights
    :type w: np.ndarray OR cp.Variable
    :param obj: objective function expression
    :type obj: cp.Expression
    :return: value of the objective function OR objective function expression
    :rtype: float OR cp.Expression
    """
    if isinstance(w, np.ndarray):
        if np.isscalar(obj):
            return obj
        elif np.isscalar(obj.value):
            return obj.value
        else:
            return obj.value.item()
    else:
        return obj


[docs]def portfolio_variance(w, cov_matrix): """ Calculate the total portfolio variance (i.e square volatility). :param w: asset weights in the portfolio :type w: np.ndarray OR cp.Variable :param cov_matrix: covariance matrix :type cov_matrix: np.ndarray :return: value of the objective function OR objective function expression :rtype: float OR cp.Expression """ variance = cp.quad_form(w, cov_matrix) return _objective_value(w, variance)
[docs]def portfolio_return(w, expected_returns, negative=True): """ Calculate the (negative) mean return of a portfolio :param w: asset weights in the portfolio :type w: np.ndarray OR cp.Variable :param expected_returns: expected return of each asset :type expected_returns: np.ndarray :param negative: whether quantity should be made negative (so we can minimise) :type negative: boolean :return: negative mean return :rtype: float """ sign = -1 if negative else 1 mu = w @ expected_returns return _objective_value(w, sign * mu)
[docs]def sharpe_ratio(w, expected_returns, cov_matrix, risk_free_rate=0.02, negative=True): """ Calculate the (negative) Sharpe ratio of a portfolio :param w: asset weights in the portfolio :type w: np.ndarray OR cp.Variable :param expected_returns: expected return of each asset :type expected_returns: np.ndarray :param cov_matrix: covariance matrix :type cov_matrix: np.ndarray :param risk_free_rate: risk-free rate of borrowing/lending, defaults to 0.02. The period of the risk-free rate should correspond to the frequency of expected returns. :type risk_free_rate: float, optional :param negative: whether quantity should be made negative (so we can minimise) :type negative: boolean :return: (negative) Sharpe ratio :rtype: float """ mu = w @ expected_returns sigma = cp.sqrt(cp.quad_form(w, cov_matrix)) sign = -1 if negative else 1 sharpe = (mu - risk_free_rate) / sigma return _objective_value(w, sign * sharpe)
[docs]def L2_reg(w, gamma=1): r""" L2 regularisation, i.e :math:`\gamma ||w||^2`, to increase the number of nonzero weights. Example:: ef = EfficientFrontier(mu, S) ef.add_objective(objective_functions.L2_reg, gamma=2) ef.min_volatility() :param w: asset weights in the portfolio :type w: np.ndarray OR cp.Variable :param gamma: L2 regularisation parameter, defaults to 1. Increase if you want more non-negligible weights :type gamma: float, optional :return: value of the objective function OR objective function expression :rtype: float OR cp.Expression """ L2_reg = gamma * cp.sum_squares(w) return _objective_value(w, L2_reg)
[docs]def quadratic_utility(w, expected_returns, cov_matrix, risk_aversion, negative=True): r""" Quadratic utility function, i.e :math:`\mu - \frac 1 2 \delta w^T \Sigma w`. :param w: asset weights in the portfolio :type w: np.ndarray OR cp.Variable :param expected_returns: expected return of each asset :type expected_returns: np.ndarray :param cov_matrix: covariance matrix :type cov_matrix: np.ndarray :param risk_aversion: risk aversion coefficient. Increase to reduce risk. :type risk_aversion: float :param negative: whether quantity should be made negative (so we can minimise). :type negative: boolean :return: value of the objective function OR objective function expression :rtype: float OR cp.Expression """ sign = -1 if negative else 1 mu = w @ expected_returns variance = cp.quad_form(w, cov_matrix) utility = mu - 0.5 * risk_aversion * variance return _objective_value(w, sign * utility)
[docs]def transaction_cost(w, w_prev, k=0.001): """ A very simple transaction cost model: sum all the weight changes and multiply by a given fraction (default to 10bps). This simulates a fixed percentage commission from your broker. :param w: asset weights in the portfolio :type w: np.ndarray OR cp.Variable :param w_prev: previous weights :type w_prev: np.ndarray :param k: fractional cost per unit weight exchanged :type k: float :return: value of the objective function OR objective function expression :rtype: float OR cp.Expression """ return _objective_value(w, k * cp.norm(w - w_prev, 1))
[docs]def ex_ante_tracking_error(w, cov_matrix, benchmark_weights): """ Calculate the (square of) the ex-ante Tracking Error, i.e :math:`(w - w_b)^T \\Sigma (w-w_b)`. :param w: asset weights in the portfolio :type w: np.ndarray OR cp.Variable :param cov_matrix: covariance matrix :type cov_matrix: np.ndarray :param benchmark_weights: asset weights in the benchmark :type benchmark_weights: np.ndarray :return: value of the objective function OR objective function expression :rtype: float OR cp.Expression """ relative_weights = w - benchmark_weights tracking_error = cp.quad_form(relative_weights, cov_matrix) return _objective_value(w, tracking_error)
[docs]def ex_post_tracking_error(w, historic_returns, benchmark_returns): """ Calculate the (square of) the ex-post Tracking Error, i.e :math:`Var(r - r_b)`. :param w: asset weights in the portfolio :type w: np.ndarray OR cp.Variable :param historic_returns: historic asset returns :type historic_returns: np.ndarray :param benchmark_returns: historic benchmark returns :type benchmark_returns: pd.Series or np.ndarray :return: value of the objective function OR objective function expression :rtype: float OR cp.Expression """ if not isinstance(historic_returns, np.ndarray): historic_returns = np.array(historic_returns) if not isinstance(benchmark_returns, np.ndarray): benchmark_returns = np.array(benchmark_returns) x_i = w @ historic_returns.T - benchmark_returns mean = cp.sum(x_i) / len(benchmark_returns) tracking_error = cp.sum_squares(x_i - mean) return _objective_value(w, tracking_error)