|
|
import numpy as np |
|
|
import pandas as pd |
|
|
|
|
|
def identify_patterns(df): |
|
|
"""Identify candlestick patterns in the data using basic calculations""" |
|
|
patterns = pd.DataFrame(index=df.index) |
|
|
|
|
|
|
|
|
body = df['Close'] - df['Open'] |
|
|
body_abs = abs(body) |
|
|
upper_shadow = df['High'] - df[['Open', 'Close']].max(axis=1) |
|
|
lower_shadow = df[['Open', 'Close']].min(axis=1) - df['Low'] |
|
|
|
|
|
|
|
|
patterns['HAMMER'] = np.where( |
|
|
(lower_shadow > 2 * body_abs) & |
|
|
(upper_shadow <= 0.1 * body_abs) & |
|
|
(body > 0), |
|
|
1, 0 |
|
|
) |
|
|
|
|
|
|
|
|
patterns['INVERTED_HAMMER'] = np.where( |
|
|
(upper_shadow > 2 * body_abs) & |
|
|
(lower_shadow <= 0.1 * body_abs) & |
|
|
(body > 0), |
|
|
1, 0 |
|
|
) |
|
|
|
|
|
|
|
|
patterns['PIERCING_LINE'] = np.where( |
|
|
(body.shift(1) < 0) & |
|
|
(body > 0) & |
|
|
(df['Open'] < df['Close'].shift(1)) & |
|
|
(df['Close'] > (df['Open'].shift(1) + df['Close'].shift(1)) / 2), |
|
|
1, 0 |
|
|
) |
|
|
|
|
|
|
|
|
patterns['BULLISH_ENGULFING'] = np.where( |
|
|
(body.shift(1) < 0) & |
|
|
(body > 0) & |
|
|
(df['Open'] < df['Close'].shift(1)) & |
|
|
(df['Close'] > df['Open'].shift(1)), |
|
|
1, 0 |
|
|
) |
|
|
|
|
|
|
|
|
patterns['MORNING_STAR'] = np.where( |
|
|
(body.shift(2) < 0) & |
|
|
(abs(body.shift(1)) < abs(body.shift(2)) * 0.3) & |
|
|
(body > 0) & |
|
|
(df['Close'] > df['Close'].shift(2) * 0.5), |
|
|
1, 0 |
|
|
) |
|
|
|
|
|
|
|
|
patterns['THREE_WHITE_SOLDIERS'] = np.where( |
|
|
(body > 0) & |
|
|
(body.shift(1) > 0) & |
|
|
(body.shift(2) > 0) & |
|
|
(df['Close'] > df['Close'].shift(1)) & |
|
|
(df['Close'].shift(1) > df['Close'].shift(2)), |
|
|
1, 0 |
|
|
) |
|
|
|
|
|
|
|
|
patterns['BULLISH_HARAMI'] = np.where( |
|
|
(body.shift(1) < 0) & |
|
|
(body > 0) & |
|
|
(df['Open'] > df['Close'].shift(1)) & |
|
|
(df['Close'] < df['Open'].shift(1)), |
|
|
1, 0 |
|
|
) |
|
|
|
|
|
|
|
|
patterns['HANGING_MAN'] = np.where( |
|
|
(lower_shadow > 2 * body_abs) & |
|
|
(upper_shadow <= 0.1 * body_abs) & |
|
|
(body < 0), |
|
|
1, 0 |
|
|
) |
|
|
|
|
|
|
|
|
patterns['DARK_CLOUD_COVER'] = np.where( |
|
|
(body.shift(1) > 0) & |
|
|
(body < 0) & |
|
|
(df['Open'] > df['High'].shift(1)) & |
|
|
(df['Close'] < (df['Open'].shift(1) + df['Close'].shift(1)) / 2), |
|
|
1, 0 |
|
|
) |
|
|
|
|
|
|
|
|
patterns['BEARISH_ENGULFING'] = np.where( |
|
|
(body.shift(1) > 0) & |
|
|
(body < 0) & |
|
|
(df['Open'] > df['Close'].shift(1)) & |
|
|
(df['Close'] < df['Open'].shift(1)), |
|
|
1, 0 |
|
|
) |
|
|
|
|
|
|
|
|
patterns['EVENING_STAR'] = np.where( |
|
|
(body.shift(2) > 0) & |
|
|
(abs(body.shift(1)) < abs(body.shift(2)) * 0.3) & |
|
|
(body < 0) & |
|
|
(df['Close'] < df['Close'].shift(2) * 0.5), |
|
|
1, 0 |
|
|
) |
|
|
|
|
|
|
|
|
patterns['THREE_BLACK_CROWS'] = np.where( |
|
|
(body < 0) & |
|
|
(body.shift(1) < 0) & |
|
|
(body.shift(2) < 0) & |
|
|
(df['Close'] < df['Close'].shift(1)) & |
|
|
(df['Close'].shift(1) < df['Close'].shift(2)), |
|
|
1, 0 |
|
|
) |
|
|
|
|
|
|
|
|
patterns['SHOOTING_STAR'] = np.where( |
|
|
(upper_shadow > 2 * body_abs) & |
|
|
(lower_shadow <= 0.1 * body_abs) & |
|
|
(body < 0), |
|
|
1, 0 |
|
|
) |
|
|
|
|
|
|
|
|
patterns['DOJI'] = np.where( |
|
|
abs(body) <= 0.1 * (df['High'] - df['Low']), |
|
|
1, 0 |
|
|
) |
|
|
|
|
|
|
|
|
patterns['DRAGONFLY_DOJI'] = np.where( |
|
|
(abs(body) <= 0.1 * (df['High'] - df['Low'])) & |
|
|
(upper_shadow <= 0.1 * (df['High'] - df['Low'])) & |
|
|
(lower_shadow >= 0.7 * (df['High'] - df['Low'])), |
|
|
1, 0 |
|
|
) |
|
|
|
|
|
|
|
|
patterns['GRAVESTONE_DOJI'] = np.where( |
|
|
(abs(body) <= 0.1 * (df['High'] - df['Low'])) & |
|
|
(lower_shadow <= 0.1 * (df['High'] - df['Low'])) & |
|
|
(upper_shadow >= 0.7 * (df['High'] - df['Low'])), |
|
|
1, 0 |
|
|
) |
|
|
|
|
|
return patterns |
|
|
|
|
|
def calculate_technical_indicators(df): |
|
|
"""Calculate technical indicators for analysis""" |
|
|
|
|
|
delta = df['Close'].diff() |
|
|
gain = (delta.where(delta > 0, 0)).rolling(window=14).mean() |
|
|
loss = (-delta.where(delta < 0, 0)).rolling(window=14).mean() |
|
|
rs = gain / loss |
|
|
df['RSI'] = 100 - (100 / (1 + rs)) |
|
|
|
|
|
|
|
|
exp1 = df['Close'].ewm(span=12, adjust=False).mean() |
|
|
exp2 = df['Close'].ewm(span=26, adjust=False).mean() |
|
|
df['MACD'] = exp1 - exp2 |
|
|
df['MACD_Signal'] = df['MACD'].ewm(span=9, adjust=False).mean() |
|
|
df['MACD_Hist'] = df['MACD'] - df['MACD_Signal'] |
|
|
|
|
|
|
|
|
df['SMA_20'] = df['Close'].rolling(window=20).mean() |
|
|
df['SMA_50'] = df['Close'].rolling(window=50).mean() |
|
|
|
|
|
return df |