Module pinkfish.indicator

Custom indicators.

These indicators are meant to supplement the TA-Lib. See: https://ta-lib.org/function.html

Expand source code
"""
Custom indicators.

These indicators are meant to supplement the TA-Lib.  See:
https://ta-lib.org/function.html
"""

import math
import numpy as np
import pandas as pd

import pinkfish.pfstatistics as pfstatistics


class IndicatorError(Exception):
    """
    Base indicator exception.
    """
    pass


########################################################################
# SMA

def SMA(ts, timeperiod=30, price='close'):
    """
    This indicator computes a simple moving average.

    Can be used in place of talib SMA.

    ts : pd.DateFrame or pd.Series
        A dataframe with 'open', 'high', 'low', 'close', 'volume' or
        a series of price data.
    timeperiod: int, optional
        The timeperiod for the moving average (default is 30).
    price : str, optional {'close', 'open', 'high', 'low'}
        Input_array column to use for price (default is 'close').
        Not used if `ts` is a series.

    Returns
    -------
    pd.Series
        Series that contains the simple moving average.

    Examples
    --------
    >>> ts['sma50'] = pf.SMA(ts, timeperiod=50)
    """
    s = ts[price] if isinstance(ts, pd.DataFrame) else ts
    return s.rolling(timeperiod).mean()


########################################################################
# EMA

def EMA(ts, timeperiod=30, price='close'):
    """
    This indicator computes an exponential moving average.

    Can be used in place of talib EMA.

    ts : pd.DateFrame or pd.Series
        A dataframe with 'open', 'high', 'low', 'close', 'volume' or
        a series of price data.
    timeperiod: int, optional
        The timeperiod for the moving average (default is 30).
    price : str, optional {'close', 'open', 'high', 'low'}
        Input_array column to use for price (default is 'close').
        Not used if `ts` is a series.

    Returns
    -------
    pd.Series
        Series that contains the simple moving average.

    Examples
    --------
    >>> ts['ema50'] = pf.EMA(ts, timeperiod=50)
    """
    s = ts[price] if isinstance(ts, pd.DataFrame) else ts
    return s.ewm(span=timeperiod, min_periods=timeperiod, adjust=False).mean()


########################################################################
# CROSSOVER

class TradeCrossOverError(IndicatorError):
    """
    Invalid timeperiod specified.
    """
    pass


class _CrossOver:
    """
    This is a helper class to implement the CROSSOVER function.

     The class provides the apply callback for pd.DataFrame.apply()
     in CROSSOVER.  It also keeps track of _r, explained below.

    _r indicates regime direction and duration, i.e. 50 means a bull
       market that has persisted for 50 days, whereas -20 means a bear
       market that has persisted for 20 days.
    _r is incremented(decremented) each day a bull(bear) market persists
    _r remains unchanged when fast_ma within band of slow_ma
    _r indicates the number of trading days a trend has persisted
    _r is nan, then sma_slow is nan
    _r > 0, then bull market, fast_ma > slow_ma
    _r < 0, then bear market, fast_ma < slow_ma
    _r == 0, no trend established yet
    """
    def __init__(self):
        """
        Initialize instance variables.

        Attributes
        ----------
        _r : int
            Indicates regime direction and duration.
        """
        self._r = 0

    def apply(self, row, band=0):
        """
        Implements the regime change logic.

        Parameters
        ----------
        row : pd.Series
            A row of data from the dataframe.
        band : int {0-100}
            Percent band (default is 0, which is no band).

        Returns
        -------
        _r : int
            Indicates regime direction and duration.
        """
        if pd.isnull(row['__sma_slow__']):
            self._r = np.nan
        elif row['__sma_fast__'] > row['__sma_slow__']*(1+band/100):
            self._r = self._r + 1 if self._r > 0 else 1
        elif row['__sma_fast__'] < row['__sma_slow__']*(1-band/100):
            self._r = self._r -1 if self._r < 0 else -1
        else:
            pass
        return self._r


def CROSSOVER(ts, timeperiod_fast=50, timeperiod_slow=200,
              func_fast=SMA, func_slow=SMA, band=0,
              price='close', prevday=False):
    """
    This indicator is used to represent regime direction and duration.

    For example, an indicator value of 50 means a bull market that has
    persisted for 50 days, whereas -20 means a bear market that has
    persisted for 20 days.

    More generally, this is a crossover indicator for two moving
    averages.  The indicator is positive when the fast moving average
    is above the slow moving arverage, and negative when the fast
    moving average is below the slow moving average.

    Parameters
    ----------
    ts : pd.DateFrame
        A dataframe with 'open', 'high', 'low', 'close', 'volume'.
    timeperiod_fast : int, optional
        The timeperiod for the fast moving average (default is 50).
    timeperiod_slow : int, optional
        The timeperiod for the slow moving average (default is 200).
    func_fast : Function, optional
        {pf.SMA, pf.EMA} (pinkfish functions) or
        {SMA, DEMA, EMA, KAMA, T3, TEMA, TRIMA, WMA} (ta-lib functions)
        The function for fast moving average (default is pf.SMA).
        MAMA not compatible.
    func_slow : Function, optional
         {pf.SMA, pf.EMA} (pinkfish functions) or
        {SMA, DEMA, EMA, KAMA, T3, TEMA, TRIMA, WMA} (ta-lib functions)
        The function for fast moving average (default is pf.SMA).
        MAMA not compatible.
    band : float, {0-100}, optional
        Percent band around the slow moving average.
        (default is 0, which implies no band is used).
    price : str, optional {'close', 'open', 'high', 'low'}
        Input_array column to use for price (default is 'close').
    prevday : bool, optional
        True will shift the series forward.  Unless you are buying
        on the close, you'll likely want to set this to True.
        It gives you the previous day's CrossOver (default is False).

    Returns
    -------
    s : pd.Series
        Series that contains the rolling regime indicator values.

    Raises
    ------
    TradeCrossOverError
        If one of the timeperiods specified is invalid.

    Examples
    --------
    >>> ts['regime'] = pf.CROSSOVER(ts, timeperiod_fast=50,
                                    timeperiod_slow=200)
    """
    if (timeperiod_fast < 1 or timeperiod_slow < 2
        or timeperiod_fast >= timeperiod_slow):
        raise TradeCrossOverError

    ts['__sma_fast__'] = ts[price] if timeperiod_fast == 1 else \
        func_fast(ts, timeperiod=timeperiod_fast, price=price)

    ts['__sma_slow__'] = \
        func_slow(ts, timeperiod=timeperiod_slow, price=price)

    func = _CrossOver().apply
    s = ts.apply(func, band=band, axis=1)
    if prevday:
        s = s.shift()
    ts.drop(['__sma_fast__', '__sma_slow__'], axis=1, inplace=True)
    return s


########################################################################
# MOMENTUM

def MOMENTUM(ts, lookback=1, time_frame='monthly', price='close', prevday=False):
    """
    This indicator is used to represent momentum is security prices.

    Percent price change is used to calculate momentum.  Momentum
    is positive if the price since the lookback period has increased.
    Likewise, if price has decreased since the lookback period,
    momentum is negative.  Percent change is used to normalize
    asset prices for comparison.

    Parameters
    ----------
    ts : pd.DateFrame
        A dataframe with 'open', 'high', 'low', 'close', 'volume'.
    lookback : int, optional
        The number of time frames to lookback, e.g. 2 months
        (default is 1).
    timeframe : str, optional {'monthly', 'daily', 'weekly', 'yearly'}
        The unit or timeframe type of lookback (default is 'monthly').
    price : str, optional {'close', 'open', 'high', 'low'}
        Input_array column to use for price (default is 'close').
    prevday : bool, optional
        True will shift the series forward.  Unless you are buying
        on the close, you'll likely want to set this to True.
        It gives you the previous day's Momentum (default is False).

    Returns
    -------
    s : pd.Series
        Series that contains the rolling momentum indicator values.

    Raises
    ------
    ValueError
        If the lookback is not positive or the time_frame is invalid.

    Examples
    --------
    >>> ts['mom'] = pf.MOMENTUM(ts, lookback=6, time_frame='monthly')
    """
    if lookback < 1:
        raise ValueError('lookback must be positive')

    if   time_frame =='daily':   factor = 1
    elif time_frame =='weekly':  factor = pfstatistics.TRADING_DAYS_PER_WEEK
    elif time_frame =='monthly': factor = pfstatistics.TRADING_DAYS_PER_MONTH
    elif time_frame =='yearly':  factor = pfstatistics.TRADING_DAYS_PER_YEAR
    else:
        raise ValueError(f'invalid time_frame "{time_frame}"')

    s = ts[price].pct_change(periods=lookback*factor)
    if prevday:
        s = s.shift()

    return s


########################################################################
# VOLATILITY

def VOLATILITY(ts, lookback=20, time_frame='yearly', downside=False,
               price='close', prevday=False):
    """
    This indicator is used to represent volatility in security prices.

    Volatility is represented as the standard deviation.  Volatility
    is calculated over the lookback period, then we scale to the
    time frame.  Volatility scales with the square root of time.
    For example,  if the market’s daily volatility is 0.5%, then
    volatility for two days is the square root of 2 times
    the daily volatility (0.5% * 1.414 = 0.707%).  We use the square
    root of time to scale from daily to weely, monthly, or yearly.

    Parameters
    ----------
    ts : pd.DateFrame
        A dataframe with 'open', 'high', 'low', 'close', 'volume'.
    lookback : int, optional
        The number of time frames to lookback, e.g. 2 months
        (default is 1).
    timeframe : str, optional {'yearly', 'daily', 'weekly', 'monthly'}
        The unit or timeframe used for scaling.  For example, if the
        lookback is 20 and the timeframe is 'yearly', then we compute
        the 20 day volatility and scale to 1 year.
        (default is 'yearly').
    downside : bool, optional
        True to calculate the downside volatility (default is False).
    price : str, optional {'close', 'open', 'high', 'low'}
        Input_array column to use for price (default is 'close').
    prevday : bool, optional
        True will shift the series forward.  Unless you are buying
        on the close, you'll likely want to set this to True.
        It gives you the previous day's Volatility (default is False).

    Returns
    -------
    s : pd.Series
        A new column that contains the rolling volatility.

    Raises
    ------
    ValueError
        If the lookback is not positive or the time_frame is invalid.

    Examples
    --------
    >>> ts['vola'] = pf.VOLATILITY(ts, lookback=20, time_frame='yearly')
    """
    if lookback < 1:
        raise ValueError('lookback must be positive')

    if   time_frame == 'daily':   factor = 1
    elif time_frame == 'weekly':  factor = pfstatistics.TRADING_DAYS_PER_WEEK
    elif time_frame == 'monthly': factor = pfstatistics.TRADING_DAYS_PER_MONTH
    elif time_frame == 'yearly':  factor = pfstatistics.TRADING_DAYS_PER_YEAR
    else:
        raise ValueError(f'invalid time_frame "{time_frame}"')

    s = ts[price].pct_change()
    if downside:
        s[s > 0] = 0
    s = s.rolling(window=lookback).std() * np.sqrt(factor)

    if prevday:
        s = s.shift()

    return s


########################################################################
# ANNUALIZED_RETURNS

def ANNUALIZED_RETURNS(ts, lookback=5, price='close', prevday=False):
    """
    Calculate the rolling annualized returns.

    Parameters
    ----------
    ts : pd.DateFrame
        A dataframe with 'open', 'high', 'low', 'close', 'volume'.
    lookback : float, optional
        The number of years to lookback, e.g. 5 years.  1/12 can be
        used for 1 month.  Likewise 3/12 for 3 months, etc...
        (default is 5).
    price : str, optional {'close', 'open', 'high', 'low'}
        Input_array column to use for price (default is 'close').
    prevday : bool, optional
        True will shift the series forward.  Unless you are buying
        on the close, you'll likely want to set this to True.
        It gives you the previous day's Volatility (default is False).

    Returns
    -------
    s : pd.Series
        Series that contains the rolling annualized returns.

    Raises
    ------
    ValueError
        If the lookback is not positive.

    Examples
    --------
    >>> annual_returns_1mo = pf.ANNUALIZED_RETURNS(ts, lookback=1/12)
    >>> annual_returns_3mo = pf.ANNUALIZED_RETURNS(ts, lookback=3/12)
    >>> annual_returns_1yr = pf.ANNUALIZED_RETURNS(ts, lookback=1)
    >>> annual_returns_3yr = pf.ANNUALIZED_RETURNS(ts, lookback=3)
    >>> annual_returns_5yr = pf.ANNUALIZED_RETURNS(ts, lookback=5)
    """
    def _cagr(s):
        """
        Calculate compound annual growth rate.

        B = end balance; A = begin balance; n = num years
        """
        A = s.iloc[0]
        B = s.iloc[-1]
        n = len(s)
        if B < 0: B = 0
        return (math.pow(B / A, 1 / n) - 1) * 100

    if lookback <= 0:
        raise ValueError('lookback must be positive')

    window = int(lookback * pfstatistics.TRADING_DAYS_PER_YEAR)
    s = pd.Series(ts[price]).rolling(window).apply(_cagr)
    if prevday:
        s = s.shift()

    return s


########################################################################
# ANNUALIZED_STANDARD_DEVIATION

def ANNUALIZED_STANDARD_DEVIATION(ts, lookback=3, price='close', prevday=False):
    """
    Calculate the rolling annualized standard deviation.

    Parameters
    ----------
    ts : pd.DateFrame
        A dataframe with 'open', 'high', 'low', 'close', 'volume'.
    lookback : float, optional
        The number of years to lookback, e.g. 5 years.  1/12 can be
        used for 1 month.  Likewise 3/12 for 3 months, etc...
        (default is 5).
    price : str, optional {'close', 'open', 'high', 'low'}
        Input_array column to use for price (default is 'close').
    prevday : bool, optional
        True will shift the series forward.  Unless you are buying
        on the close, you'll likely want to set this to True.
        It gives you the previous day's Volatility (default is False).

    Returns
    -------
    s : pd.Series
        Series that contains the rolling annualized standard deviation.

    Raises
    ------
    ValueError
        If the lookback is not positive.

    Examples
    --------
    >>> std_dev_1mo = pf.ANNUALIZED_STANDARD_DEVIATION(ts,lookback=1/12)
    >>> std_dev_3mo = pf.ANNUALIZED_STANDARD_DEVIATION(ts, lookback=3/12)
    >>> std_dev_1yr = pf.ANNUALIZED_STANDARD_DEVIATION(ts, lookback=1)
    >>> std_dev_3yr = pf.ANNUALIZED_STANDARD_DEVIATION(ts, lookback=3)
    >>> std_dev_5yr = pf.ANNUALIZED_STANDARD_DEVIATION(ts, lookback=5)
    """
    def _std_dev(s):
        """
        Calculate the annualized standard deviation.
        """
        return np.std(s, axis=0) * math.sqrt(pfstatistics.TRADING_DAYS_PER_YEAR)

    if lookback <= 0:
        raise ValueError('lookback must be positive')

    window = int(lookback * pfstatistics.TRADING_DAYS_PER_YEAR)
    pc = ts[price].pct_change()
    s = pd.Series(pc).rolling(window).apply(_std_dev)
    if prevday:
        s = s.shift()

    return s


########################################################################
# ANNUALIZED_SHARPE_RATIO

def ANNUALIZED_SHARPE_RATIO(ts, lookback=5, price='close', prevday=False,
                            risk_free=0):
    """
    Calculate the rolling annualized sharpe ratio.

    Parameters
    ----------
    ts : pd.DateFrame
        A dataframe with 'open', 'high', 'low', 'close', 'volume'.
    lookback : float, optional
        The number of years to lookback, e.g. 5 years.  1/12 can be
        used for 1 month.  Likewise 3/12 for 3 months, etc...
        (default is 5).
    price : str, optional {'close', 'open', 'high', 'low'}
        Input_array column to use for price (default is 'close').
    prevday : bool, optional
        True will shift the series forward.  Unless you are buying
        on the close, you'll likely want to set this to True.
        It gives you the previous day's Volatility (default is False).
    risk_free: float, optional
        The risk free rate (default is 0).

    Returns
    -------
    s : pd.Series
        Series that contains the rolling annualized sharpe ratio.

    Raises
    ------
    ValueError
        If the lookback is not positive.

    Examples
    --------
    >>> sharpe_ratio_1mo = pf.ANNUALIZED_SHARPE_RATIO(ts, lookback=1/12)
    >>> sharpe_ratio_3mo = pf.ANNUALIZED_SHARPE_RATIO(ts, lookback=3/12)
    >>> sharpe_ratio_1yr = pf.ANNUALIZED_SHARPE_RATIO(ts, lookback=1)
    >>> sharpe_ratio_3yr = pf.ANNUALIZED_SHARPE_RATIO(ts, lookback=3)
    >>> sharpe_ratio_5yr = pf.ANNUALIZED_SHARPE_RATIO(ts, lookback=5)
    """
    def _sharpe_ratio(s):
        """
        Calculate the annualized sharpe ratio.
        """
        dev = np.std(s, axis=0)
        mean = np.mean(s, axis=0)
        period = len(s)
        sharpe = (mean*period - risk_free) / (dev * np.sqrt(period))
        return sharpe

    if lookback <= 0:
        raise ValueError('lookback must be positive')

    window = int(lookback * pfstatistics.TRADING_DAYS_PER_YEAR)
    pc = ts[price].pct_change()
    s = pd.Series(pc).rolling(window).apply(_sharpe_ratio)
    if prevday:
        s = s.shift()

    return s

Functions

def ANNUALIZED_RETURNS(ts, lookback=5, price='close', prevday=False)

Calculate the rolling annualized returns.

Parameters

ts : pd.DateFrame
A dataframe with 'open', 'high', 'low', 'close', 'volume'.
lookback : float, optional
The number of years to lookback, e.g. 5 years. 1/12 can be used for 1 month. Likewise 3/12 for 3 months, etc… (default is 5).
price : str, optional {'close', 'open', 'high', 'low'}
Input_array column to use for price (default is 'close').
prevday : bool, optional
True will shift the series forward. Unless you are buying on the close, you'll likely want to set this to True. It gives you the previous day's Volatility (default is False).

Returns

s : pd.Series
Series that contains the rolling annualized returns.

Raises

ValueError
If the lookback is not positive.

Examples

>>> annual_returns_1mo = pf.ANNUALIZED_RETURNS(ts, lookback=1/12)
>>> annual_returns_3mo = pf.ANNUALIZED_RETURNS(ts, lookback=3/12)
>>> annual_returns_1yr = pf.ANNUALIZED_RETURNS(ts, lookback=1)
>>> annual_returns_3yr = pf.ANNUALIZED_RETURNS(ts, lookback=3)
>>> annual_returns_5yr = pf.ANNUALIZED_RETURNS(ts, lookback=5)
Expand source code
def ANNUALIZED_RETURNS(ts, lookback=5, price='close', prevday=False):
    """
    Calculate the rolling annualized returns.

    Parameters
    ----------
    ts : pd.DateFrame
        A dataframe with 'open', 'high', 'low', 'close', 'volume'.
    lookback : float, optional
        The number of years to lookback, e.g. 5 years.  1/12 can be
        used for 1 month.  Likewise 3/12 for 3 months, etc...
        (default is 5).
    price : str, optional {'close', 'open', 'high', 'low'}
        Input_array column to use for price (default is 'close').
    prevday : bool, optional
        True will shift the series forward.  Unless you are buying
        on the close, you'll likely want to set this to True.
        It gives you the previous day's Volatility (default is False).

    Returns
    -------
    s : pd.Series
        Series that contains the rolling annualized returns.

    Raises
    ------
    ValueError
        If the lookback is not positive.

    Examples
    --------
    >>> annual_returns_1mo = pf.ANNUALIZED_RETURNS(ts, lookback=1/12)
    >>> annual_returns_3mo = pf.ANNUALIZED_RETURNS(ts, lookback=3/12)
    >>> annual_returns_1yr = pf.ANNUALIZED_RETURNS(ts, lookback=1)
    >>> annual_returns_3yr = pf.ANNUALIZED_RETURNS(ts, lookback=3)
    >>> annual_returns_5yr = pf.ANNUALIZED_RETURNS(ts, lookback=5)
    """
    def _cagr(s):
        """
        Calculate compound annual growth rate.

        B = end balance; A = begin balance; n = num years
        """
        A = s.iloc[0]
        B = s.iloc[-1]
        n = len(s)
        if B < 0: B = 0
        return (math.pow(B / A, 1 / n) - 1) * 100

    if lookback <= 0:
        raise ValueError('lookback must be positive')

    window = int(lookback * pfstatistics.TRADING_DAYS_PER_YEAR)
    s = pd.Series(ts[price]).rolling(window).apply(_cagr)
    if prevday:
        s = s.shift()

    return s
def ANNUALIZED_SHARPE_RATIO(ts, lookback=5, price='close', prevday=False, risk_free=0)

Calculate the rolling annualized sharpe ratio.

Parameters

ts : pd.DateFrame
A dataframe with 'open', 'high', 'low', 'close', 'volume'.
lookback : float, optional
The number of years to lookback, e.g. 5 years. 1/12 can be used for 1 month. Likewise 3/12 for 3 months, etc… (default is 5).
price : str, optional {'close', 'open', 'high', 'low'}
Input_array column to use for price (default is 'close').
prevday : bool, optional
True will shift the series forward. Unless you are buying on the close, you'll likely want to set this to True. It gives you the previous day's Volatility (default is False).
risk_free : float, optional
The risk free rate (default is 0).

Returns

s : pd.Series
Series that contains the rolling annualized sharpe ratio.

Raises

ValueError
If the lookback is not positive.

Examples

>>> sharpe_ratio_1mo = pf.ANNUALIZED_SHARPE_RATIO(ts, lookback=1/12)
>>> sharpe_ratio_3mo = pf.ANNUALIZED_SHARPE_RATIO(ts, lookback=3/12)
>>> sharpe_ratio_1yr = pf.ANNUALIZED_SHARPE_RATIO(ts, lookback=1)
>>> sharpe_ratio_3yr = pf.ANNUALIZED_SHARPE_RATIO(ts, lookback=3)
>>> sharpe_ratio_5yr = pf.ANNUALIZED_SHARPE_RATIO(ts, lookback=5)
Expand source code
def ANNUALIZED_SHARPE_RATIO(ts, lookback=5, price='close', prevday=False,
                            risk_free=0):
    """
    Calculate the rolling annualized sharpe ratio.

    Parameters
    ----------
    ts : pd.DateFrame
        A dataframe with 'open', 'high', 'low', 'close', 'volume'.
    lookback : float, optional
        The number of years to lookback, e.g. 5 years.  1/12 can be
        used for 1 month.  Likewise 3/12 for 3 months, etc...
        (default is 5).
    price : str, optional {'close', 'open', 'high', 'low'}
        Input_array column to use for price (default is 'close').
    prevday : bool, optional
        True will shift the series forward.  Unless you are buying
        on the close, you'll likely want to set this to True.
        It gives you the previous day's Volatility (default is False).
    risk_free: float, optional
        The risk free rate (default is 0).

    Returns
    -------
    s : pd.Series
        Series that contains the rolling annualized sharpe ratio.

    Raises
    ------
    ValueError
        If the lookback is not positive.

    Examples
    --------
    >>> sharpe_ratio_1mo = pf.ANNUALIZED_SHARPE_RATIO(ts, lookback=1/12)
    >>> sharpe_ratio_3mo = pf.ANNUALIZED_SHARPE_RATIO(ts, lookback=3/12)
    >>> sharpe_ratio_1yr = pf.ANNUALIZED_SHARPE_RATIO(ts, lookback=1)
    >>> sharpe_ratio_3yr = pf.ANNUALIZED_SHARPE_RATIO(ts, lookback=3)
    >>> sharpe_ratio_5yr = pf.ANNUALIZED_SHARPE_RATIO(ts, lookback=5)
    """
    def _sharpe_ratio(s):
        """
        Calculate the annualized sharpe ratio.
        """
        dev = np.std(s, axis=0)
        mean = np.mean(s, axis=0)
        period = len(s)
        sharpe = (mean*period - risk_free) / (dev * np.sqrt(period))
        return sharpe

    if lookback <= 0:
        raise ValueError('lookback must be positive')

    window = int(lookback * pfstatistics.TRADING_DAYS_PER_YEAR)
    pc = ts[price].pct_change()
    s = pd.Series(pc).rolling(window).apply(_sharpe_ratio)
    if prevday:
        s = s.shift()

    return s
def ANNUALIZED_STANDARD_DEVIATION(ts, lookback=3, price='close', prevday=False)

Calculate the rolling annualized standard deviation.

Parameters

ts : pd.DateFrame
A dataframe with 'open', 'high', 'low', 'close', 'volume'.
lookback : float, optional
The number of years to lookback, e.g. 5 years. 1/12 can be used for 1 month. Likewise 3/12 for 3 months, etc… (default is 5).
price : str, optional {'close', 'open', 'high', 'low'}
Input_array column to use for price (default is 'close').
prevday : bool, optional
True will shift the series forward. Unless you are buying on the close, you'll likely want to set this to True. It gives you the previous day's Volatility (default is False).

Returns

s : pd.Series
Series that contains the rolling annualized standard deviation.

Raises

ValueError
If the lookback is not positive.

Examples

>>> std_dev_1mo = pf.ANNUALIZED_STANDARD_DEVIATION(ts,lookback=1/12)
>>> std_dev_3mo = pf.ANNUALIZED_STANDARD_DEVIATION(ts, lookback=3/12)
>>> std_dev_1yr = pf.ANNUALIZED_STANDARD_DEVIATION(ts, lookback=1)
>>> std_dev_3yr = pf.ANNUALIZED_STANDARD_DEVIATION(ts, lookback=3)
>>> std_dev_5yr = pf.ANNUALIZED_STANDARD_DEVIATION(ts, lookback=5)
Expand source code
def ANNUALIZED_STANDARD_DEVIATION(ts, lookback=3, price='close', prevday=False):
    """
    Calculate the rolling annualized standard deviation.

    Parameters
    ----------
    ts : pd.DateFrame
        A dataframe with 'open', 'high', 'low', 'close', 'volume'.
    lookback : float, optional
        The number of years to lookback, e.g. 5 years.  1/12 can be
        used for 1 month.  Likewise 3/12 for 3 months, etc...
        (default is 5).
    price : str, optional {'close', 'open', 'high', 'low'}
        Input_array column to use for price (default is 'close').
    prevday : bool, optional
        True will shift the series forward.  Unless you are buying
        on the close, you'll likely want to set this to True.
        It gives you the previous day's Volatility (default is False).

    Returns
    -------
    s : pd.Series
        Series that contains the rolling annualized standard deviation.

    Raises
    ------
    ValueError
        If the lookback is not positive.

    Examples
    --------
    >>> std_dev_1mo = pf.ANNUALIZED_STANDARD_DEVIATION(ts,lookback=1/12)
    >>> std_dev_3mo = pf.ANNUALIZED_STANDARD_DEVIATION(ts, lookback=3/12)
    >>> std_dev_1yr = pf.ANNUALIZED_STANDARD_DEVIATION(ts, lookback=1)
    >>> std_dev_3yr = pf.ANNUALIZED_STANDARD_DEVIATION(ts, lookback=3)
    >>> std_dev_5yr = pf.ANNUALIZED_STANDARD_DEVIATION(ts, lookback=5)
    """
    def _std_dev(s):
        """
        Calculate the annualized standard deviation.
        """
        return np.std(s, axis=0) * math.sqrt(pfstatistics.TRADING_DAYS_PER_YEAR)

    if lookback <= 0:
        raise ValueError('lookback must be positive')

    window = int(lookback * pfstatistics.TRADING_DAYS_PER_YEAR)
    pc = ts[price].pct_change()
    s = pd.Series(pc).rolling(window).apply(_std_dev)
    if prevday:
        s = s.shift()

    return s
def CROSSOVER(ts, timeperiod_fast=50, timeperiod_slow=200, func_fast=<function SMA>, func_slow=<function SMA>, band=0, price='close', prevday=False)

This indicator is used to represent regime direction and duration.

For example, an indicator value of 50 means a bull market that has persisted for 50 days, whereas -20 means a bear market that has persisted for 20 days.

More generally, this is a crossover indicator for two moving averages. The indicator is positive when the fast moving average is above the slow moving arverage, and negative when the fast moving average is below the slow moving average.

Parameters

ts : pd.DateFrame
A dataframe with 'open', 'high', 'low', 'close', 'volume'.
timeperiod_fast : int, optional
The timeperiod for the fast moving average (default is 50).
timeperiod_slow : int, optional
The timeperiod for the slow moving average (default is 200).
func_fast : Function, optional
{pf.SMA, pf.EMA} (pinkfish functions) or {SMA, DEMA, EMA, KAMA, T3, TEMA, TRIMA, WMA} (ta-lib functions) The function for fast moving average (default is pf.SMA). MAMA not compatible.
func_slow : Function, optional
{pf.SMA, pf.EMA} (pinkfish functions) or {SMA, DEMA, EMA, KAMA, T3, TEMA, TRIMA, WMA} (ta-lib functions) The function for fast moving average (default is pf.SMA). MAMA not compatible.
band : float, {0-100}, optional
Percent band around the slow moving average. (default is 0, which implies no band is used).
price : str, optional {'close', 'open', 'high', 'low'}
Input_array column to use for price (default is 'close').
prevday : bool, optional
True will shift the series forward. Unless you are buying on the close, you'll likely want to set this to True. It gives you the previous day's CrossOver (default is False).

Returns

s : pd.Series
Series that contains the rolling regime indicator values.

Raises

TradeCrossOverError
If one of the timeperiods specified is invalid.

Examples

>>> ts['regime'] = pf.CROSSOVER(ts, timeperiod_fast=50,
                                timeperiod_slow=200)
Expand source code
def CROSSOVER(ts, timeperiod_fast=50, timeperiod_slow=200,
              func_fast=SMA, func_slow=SMA, band=0,
              price='close', prevday=False):
    """
    This indicator is used to represent regime direction and duration.

    For example, an indicator value of 50 means a bull market that has
    persisted for 50 days, whereas -20 means a bear market that has
    persisted for 20 days.

    More generally, this is a crossover indicator for two moving
    averages.  The indicator is positive when the fast moving average
    is above the slow moving arverage, and negative when the fast
    moving average is below the slow moving average.

    Parameters
    ----------
    ts : pd.DateFrame
        A dataframe with 'open', 'high', 'low', 'close', 'volume'.
    timeperiod_fast : int, optional
        The timeperiod for the fast moving average (default is 50).
    timeperiod_slow : int, optional
        The timeperiod for the slow moving average (default is 200).
    func_fast : Function, optional
        {pf.SMA, pf.EMA} (pinkfish functions) or
        {SMA, DEMA, EMA, KAMA, T3, TEMA, TRIMA, WMA} (ta-lib functions)
        The function for fast moving average (default is pf.SMA).
        MAMA not compatible.
    func_slow : Function, optional
         {pf.SMA, pf.EMA} (pinkfish functions) or
        {SMA, DEMA, EMA, KAMA, T3, TEMA, TRIMA, WMA} (ta-lib functions)
        The function for fast moving average (default is pf.SMA).
        MAMA not compatible.
    band : float, {0-100}, optional
        Percent band around the slow moving average.
        (default is 0, which implies no band is used).
    price : str, optional {'close', 'open', 'high', 'low'}
        Input_array column to use for price (default is 'close').
    prevday : bool, optional
        True will shift the series forward.  Unless you are buying
        on the close, you'll likely want to set this to True.
        It gives you the previous day's CrossOver (default is False).

    Returns
    -------
    s : pd.Series
        Series that contains the rolling regime indicator values.

    Raises
    ------
    TradeCrossOverError
        If one of the timeperiods specified is invalid.

    Examples
    --------
    >>> ts['regime'] = pf.CROSSOVER(ts, timeperiod_fast=50,
                                    timeperiod_slow=200)
    """
    if (timeperiod_fast < 1 or timeperiod_slow < 2
        or timeperiod_fast >= timeperiod_slow):
        raise TradeCrossOverError

    ts['__sma_fast__'] = ts[price] if timeperiod_fast == 1 else \
        func_fast(ts, timeperiod=timeperiod_fast, price=price)

    ts['__sma_slow__'] = \
        func_slow(ts, timeperiod=timeperiod_slow, price=price)

    func = _CrossOver().apply
    s = ts.apply(func, band=band, axis=1)
    if prevday:
        s = s.shift()
    ts.drop(['__sma_fast__', '__sma_slow__'], axis=1, inplace=True)
    return s
def EMA(ts, timeperiod=30, price='close')

This indicator computes an exponential moving average.

Can be used in place of talib EMA.

ts : pd.DateFrame or pd.Series A dataframe with 'open', 'high', 'low', 'close', 'volume' or a series of price data. timeperiod: int, optional The timeperiod for the moving average (default is 30). price : str, optional {'close', 'open', 'high', 'low'} Input_array column to use for price (default is 'close'). Not used if ts is a series.

Returns

pd.Series
Series that contains the simple moving average.

Examples

>>> ts['ema50'] = pf.EMA(ts, timeperiod=50)
Expand source code
def EMA(ts, timeperiod=30, price='close'):
    """
    This indicator computes an exponential moving average.

    Can be used in place of talib EMA.

    ts : pd.DateFrame or pd.Series
        A dataframe with 'open', 'high', 'low', 'close', 'volume' or
        a series of price data.
    timeperiod: int, optional
        The timeperiod for the moving average (default is 30).
    price : str, optional {'close', 'open', 'high', 'low'}
        Input_array column to use for price (default is 'close').
        Not used if `ts` is a series.

    Returns
    -------
    pd.Series
        Series that contains the simple moving average.

    Examples
    --------
    >>> ts['ema50'] = pf.EMA(ts, timeperiod=50)
    """
    s = ts[price] if isinstance(ts, pd.DataFrame) else ts
    return s.ewm(span=timeperiod, min_periods=timeperiod, adjust=False).mean()
def MOMENTUM(ts, lookback=1, time_frame='monthly', price='close', prevday=False)

This indicator is used to represent momentum is security prices.

Percent price change is used to calculate momentum. Momentum is positive if the price since the lookback period has increased. Likewise, if price has decreased since the lookback period, momentum is negative. Percent change is used to normalize asset prices for comparison.

Parameters

ts : pd.DateFrame
A dataframe with 'open', 'high', 'low', 'close', 'volume'.
lookback : int, optional
The number of time frames to lookback, e.g. 2 months (default is 1).
timeframe : str, optional {'monthly', 'daily', 'weekly', 'yearly'}
The unit or timeframe type of lookback (default is 'monthly').
price : str, optional {'close', 'open', 'high', 'low'}
Input_array column to use for price (default is 'close').
prevday : bool, optional
True will shift the series forward. Unless you are buying on the close, you'll likely want to set this to True. It gives you the previous day's Momentum (default is False).

Returns

s : pd.Series
Series that contains the rolling momentum indicator values.

Raises

ValueError
If the lookback is not positive or the time_frame is invalid.

Examples

>>> ts['mom'] = pf.MOMENTUM(ts, lookback=6, time_frame='monthly')
Expand source code
def MOMENTUM(ts, lookback=1, time_frame='monthly', price='close', prevday=False):
    """
    This indicator is used to represent momentum is security prices.

    Percent price change is used to calculate momentum.  Momentum
    is positive if the price since the lookback period has increased.
    Likewise, if price has decreased since the lookback period,
    momentum is negative.  Percent change is used to normalize
    asset prices for comparison.

    Parameters
    ----------
    ts : pd.DateFrame
        A dataframe with 'open', 'high', 'low', 'close', 'volume'.
    lookback : int, optional
        The number of time frames to lookback, e.g. 2 months
        (default is 1).
    timeframe : str, optional {'monthly', 'daily', 'weekly', 'yearly'}
        The unit or timeframe type of lookback (default is 'monthly').
    price : str, optional {'close', 'open', 'high', 'low'}
        Input_array column to use for price (default is 'close').
    prevday : bool, optional
        True will shift the series forward.  Unless you are buying
        on the close, you'll likely want to set this to True.
        It gives you the previous day's Momentum (default is False).

    Returns
    -------
    s : pd.Series
        Series that contains the rolling momentum indicator values.

    Raises
    ------
    ValueError
        If the lookback is not positive or the time_frame is invalid.

    Examples
    --------
    >>> ts['mom'] = pf.MOMENTUM(ts, lookback=6, time_frame='monthly')
    """
    if lookback < 1:
        raise ValueError('lookback must be positive')

    if   time_frame =='daily':   factor = 1
    elif time_frame =='weekly':  factor = pfstatistics.TRADING_DAYS_PER_WEEK
    elif time_frame =='monthly': factor = pfstatistics.TRADING_DAYS_PER_MONTH
    elif time_frame =='yearly':  factor = pfstatistics.TRADING_DAYS_PER_YEAR
    else:
        raise ValueError(f'invalid time_frame "{time_frame}"')

    s = ts[price].pct_change(periods=lookback*factor)
    if prevday:
        s = s.shift()

    return s
def SMA(ts, timeperiod=30, price='close')

This indicator computes a simple moving average.

Can be used in place of talib SMA.

ts : pd.DateFrame or pd.Series A dataframe with 'open', 'high', 'low', 'close', 'volume' or a series of price data. timeperiod: int, optional The timeperiod for the moving average (default is 30). price : str, optional {'close', 'open', 'high', 'low'} Input_array column to use for price (default is 'close'). Not used if ts is a series.

Returns

pd.Series
Series that contains the simple moving average.

Examples

>>> ts['sma50'] = pf.SMA(ts, timeperiod=50)
Expand source code
def SMA(ts, timeperiod=30, price='close'):
    """
    This indicator computes a simple moving average.

    Can be used in place of talib SMA.

    ts : pd.DateFrame or pd.Series
        A dataframe with 'open', 'high', 'low', 'close', 'volume' or
        a series of price data.
    timeperiod: int, optional
        The timeperiod for the moving average (default is 30).
    price : str, optional {'close', 'open', 'high', 'low'}
        Input_array column to use for price (default is 'close').
        Not used if `ts` is a series.

    Returns
    -------
    pd.Series
        Series that contains the simple moving average.

    Examples
    --------
    >>> ts['sma50'] = pf.SMA(ts, timeperiod=50)
    """
    s = ts[price] if isinstance(ts, pd.DataFrame) else ts
    return s.rolling(timeperiod).mean()
def VOLATILITY(ts, lookback=20, time_frame='yearly', downside=False, price='close', prevday=False)

This indicator is used to represent volatility in security prices.

Volatility is represented as the standard deviation. Volatility is calculated over the lookback period, then we scale to the time frame. Volatility scales with the square root of time. For example, if the market’s daily volatility is 0.5%, then volatility for two days is the square root of 2 times the daily volatility (0.5% * 1.414 = 0.707%). We use the square root of time to scale from daily to weely, monthly, or yearly.

Parameters

ts : pd.DateFrame
A dataframe with 'open', 'high', 'low', 'close', 'volume'.
lookback : int, optional
The number of time frames to lookback, e.g. 2 months (default is 1).
timeframe : str, optional {'yearly', 'daily', 'weekly', 'monthly'}
The unit or timeframe used for scaling. For example, if the lookback is 20 and the timeframe is 'yearly', then we compute the 20 day volatility and scale to 1 year. (default is 'yearly').
downside : bool, optional
True to calculate the downside volatility (default is False).
price : str, optional {'close', 'open', 'high', 'low'}
Input_array column to use for price (default is 'close').
prevday : bool, optional
True will shift the series forward. Unless you are buying on the close, you'll likely want to set this to True. It gives you the previous day's Volatility (default is False).

Returns

s : pd.Series
A new column that contains the rolling volatility.

Raises

ValueError
If the lookback is not positive or the time_frame is invalid.

Examples

>>> ts['vola'] = pf.VOLATILITY(ts, lookback=20, time_frame='yearly')
Expand source code
def VOLATILITY(ts, lookback=20, time_frame='yearly', downside=False,
               price='close', prevday=False):
    """
    This indicator is used to represent volatility in security prices.

    Volatility is represented as the standard deviation.  Volatility
    is calculated over the lookback period, then we scale to the
    time frame.  Volatility scales with the square root of time.
    For example,  if the market’s daily volatility is 0.5%, then
    volatility for two days is the square root of 2 times
    the daily volatility (0.5% * 1.414 = 0.707%).  We use the square
    root of time to scale from daily to weely, monthly, or yearly.

    Parameters
    ----------
    ts : pd.DateFrame
        A dataframe with 'open', 'high', 'low', 'close', 'volume'.
    lookback : int, optional
        The number of time frames to lookback, e.g. 2 months
        (default is 1).
    timeframe : str, optional {'yearly', 'daily', 'weekly', 'monthly'}
        The unit or timeframe used for scaling.  For example, if the
        lookback is 20 and the timeframe is 'yearly', then we compute
        the 20 day volatility and scale to 1 year.
        (default is 'yearly').
    downside : bool, optional
        True to calculate the downside volatility (default is False).
    price : str, optional {'close', 'open', 'high', 'low'}
        Input_array column to use for price (default is 'close').
    prevday : bool, optional
        True will shift the series forward.  Unless you are buying
        on the close, you'll likely want to set this to True.
        It gives you the previous day's Volatility (default is False).

    Returns
    -------
    s : pd.Series
        A new column that contains the rolling volatility.

    Raises
    ------
    ValueError
        If the lookback is not positive or the time_frame is invalid.

    Examples
    --------
    >>> ts['vola'] = pf.VOLATILITY(ts, lookback=20, time_frame='yearly')
    """
    if lookback < 1:
        raise ValueError('lookback must be positive')

    if   time_frame == 'daily':   factor = 1
    elif time_frame == 'weekly':  factor = pfstatistics.TRADING_DAYS_PER_WEEK
    elif time_frame == 'monthly': factor = pfstatistics.TRADING_DAYS_PER_MONTH
    elif time_frame == 'yearly':  factor = pfstatistics.TRADING_DAYS_PER_YEAR
    else:
        raise ValueError(f'invalid time_frame "{time_frame}"')

    s = ts[price].pct_change()
    if downside:
        s[s > 0] = 0
    s = s.rolling(window=lookback).std() * np.sqrt(factor)

    if prevday:
        s = s.shift()

    return s

Classes

class IndicatorError (*args, **kwargs)

Base indicator exception.

Expand source code
class IndicatorError(Exception):
    """
    Base indicator exception.
    """
    pass

Ancestors

  • builtins.Exception
  • builtins.BaseException

Subclasses

class TradeCrossOverError (*args, **kwargs)

Invalid timeperiod specified.

Expand source code
class TradeCrossOverError(IndicatorError):
    """
    Invalid timeperiod specified.
    """
    pass

Ancestors