Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions .Jules/palette.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,7 @@
## 2024-05-22 - Visual Hierarchy in CLI Output
**Learning:** Adding color-coded indicators (Green/Red) and emojis (💰, 📉) in CLI tools significantly reduces cognitive load when parsing financial data streams. It transforms a wall of text into a scannable narrative.
**Action:** For data-heavy CLI applications, always implement a semantic color system and visual anchors (icons/emojis) for key events.

## 2024-05-23 - CLI Accessibility and Configuration
**Learning:** Hardcoded simulation parameters create friction and limit accessibility. Adding CLI arguments (`argparse`) empowers users to explore scenarios without code edits. Additionally, providing a `--no-color` flag is crucial for users with visual impairments or those piping output to logs.
**Action:** Always implement standard CLI flags for configuration and accessibility (quiet mode, no-color) in command-line tools.
5 changes: 5 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -39,3 +39,8 @@

# debug information files
*.dwo

# Python
__pycache__/
*.py[cod]
*$py.class
54 changes: 45 additions & 9 deletions bitcoin_trading_simulation.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import argparse
import numpy as np
import pandas as pd

Expand All @@ -9,6 +10,15 @@ class Colors:
ENDC = '\033[0m'
BOLD = '\033[1m'

@classmethod
def disable(cls):
cls.HEADER = ''
cls.BLUE = ''
cls.GREEN = ''
cls.RED = ''
cls.ENDC = ''
cls.BOLD = ''

def simulate_bitcoin_prices(days=60, initial_price=50000, volatility=0.02):
"""
Simulates Bitcoin prices for a given number of days using Geometric Brownian Motion.
Expand Down Expand Up @@ -49,7 +59,7 @@ def generate_trading_signals(signals):
signals['positions'] = signals['signal'].diff().shift(1)
return signals

def simulate_trading(signals, initial_cash=10000):
def simulate_trading(signals, initial_cash=10000, quiet=False):
"""
Simulates trading based on signals and prints a daily ledger.
"""
Expand All @@ -59,7 +69,8 @@ def simulate_trading(signals, initial_cash=10000):
portfolio['btc'] = 0.0
portfolio['total_value'] = float(initial_cash)

print(f"{Colors.HEADER}{Colors.BOLD}------ Daily Trading Ledger ------{Colors.ENDC}")
if not quiet:
print(f"{Colors.HEADER}{Colors.BOLD}------ Daily Trading Ledger ------{Colors.ENDC}")
for i, row in signals.iterrows():
if i > 0:
portfolio.loc[i, 'cash'] = portfolio.loc[i-1, 'cash']
Expand All @@ -70,24 +81,30 @@ def simulate_trading(signals, initial_cash=10000):
btc_to_buy = portfolio.loc[i, 'cash'] / row['price']
portfolio.loc[i, 'btc'] += btc_to_buy
portfolio.loc[i, 'cash'] -= btc_to_buy * row['price']
print(f"{Colors.GREEN}Day {i}: 💰 Buy {btc_to_buy:.4f} BTC at ${row['price']:.2f}{Colors.ENDC}")
if not quiet:
print(f"{Colors.GREEN}Day {i}: 💰 Buy {btc_to_buy:.4f} BTC at ${row['price']:.2f}{Colors.ENDC}")

# Sell signal
elif row['positions'] == -2.0:
if portfolio.loc[i, 'btc'] > 0:
cash_received = portfolio.loc[i, 'btc'] * row['price']
portfolio.loc[i, 'cash'] += cash_received
print(f"{Colors.RED}Day {i}: 📉 Sell {portfolio.loc[i, 'btc']:.4f} BTC at ${row['price']:.2f}{Colors.ENDC}")
if not quiet:
print(f"{Colors.RED}Day {i}: 📉 Sell {portfolio.loc[i, 'btc']:.4f} BTC at ${row['price']:.2f}{Colors.ENDC}")
portfolio.loc[i, 'btc'] = 0

portfolio.loc[i, 'total_value'] = portfolio.loc[i, 'cash'] + portfolio.loc[i, 'btc'] * row['price']
print(f"Day {i}: Portfolio Value: ${portfolio.loc[i, 'total_value']:.2f}, Cash: ${portfolio.loc[i, 'cash']:.2f}, BTC: {portfolio.loc[i, 'btc']:.4f}")
if not quiet:
print(f"Day {i}: Portfolio Value: ${portfolio.loc[i, 'total_value']:.2f}, Cash: ${portfolio.loc[i, 'cash']:.2f}, BTC: {portfolio.loc[i, 'btc']:.4f}")

return portfolio

if __name__ == "__main__":
def main(days, initial_price, volatility, initial_cash, quiet, no_color):
if no_color:
Colors.disable()

# Simulate prices
prices = simulate_bitcoin_prices()
prices = simulate_bitcoin_prices(days=days, initial_price=initial_price, volatility=volatility)

# Calculate moving averages
signals = calculate_moving_averages(prices)
Expand All @@ -96,11 +113,10 @@ def simulate_trading(signals, initial_cash=10000):
signals = generate_trading_signals(signals)

# Simulate trading
portfolio = simulate_trading(signals)
portfolio = simulate_trading(signals, initial_cash=initial_cash, quiet=quiet)

# Final portfolio performance
final_value = portfolio['total_value'].iloc[-1]
initial_cash = 10000
profit = final_value - initial_cash

# Compare with buy and hold strategy
Expand All @@ -118,3 +134,23 @@ def simulate_trading(signals, initial_cash=10000):

print(f"Buy and Hold Strategy Value: ${buy_and_hold_value:.2f}")
print(f"{Colors.HEADER}-----------------------------------------{Colors.ENDC}")

if __name__ == "__main__":
parser = argparse.ArgumentParser(description='Bitcoin Trading Simulation')
parser.add_argument('--days', type=int, default=60, help='Number of days to simulate')
parser.add_argument('--initial-cash', type=float, default=10000, help='Initial cash amount')
parser.add_argument('--initial-price', type=float, default=50000, help='Initial Bitcoin price')
parser.add_argument('--volatility', type=float, default=0.02, help='Volatility factor')
parser.add_argument('--quiet', action='store_true', help='Suppress daily output')
parser.add_argument('--no-color', action='store_true', help='Disable colored output')

args = parser.parse_args()

main(
days=args.days,
initial_price=args.initial_price,
volatility=args.volatility,
initial_cash=args.initial_cash,
quiet=args.quiet,
no_color=args.no_color
)
64 changes: 64 additions & 0 deletions test_bitcoin_trading.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
import unittest
from unittest.mock import patch
import pandas as pd
from bitcoin_trading_simulation import Colors, simulate_trading, main, simulate_bitcoin_prices, calculate_moving_averages, generate_trading_signals

class TestBitcoinSimulation(unittest.TestCase):

def tearDown(self):
# Reset Colors to default just in case
Colors.HEADER = '\033[95m'
Colors.BLUE = '\033[94m'
Colors.GREEN = '\033[92m'
Colors.RED = '\033[91m'
Colors.ENDC = '\033[0m'
Colors.BOLD = '\033[1m'

def test_colors_disable(self):
# Disable colors
Colors.disable()

# Verify empty strings
self.assertEqual(Colors.HEADER, '')
self.assertEqual(Colors.BLUE, '')
self.assertEqual(Colors.GREEN, '')
self.assertEqual(Colors.RED, '')
self.assertEqual(Colors.ENDC, '')
self.assertEqual(Colors.BOLD, '')

@patch('builtins.print')
def test_simulate_trading_quiet(self, mock_print):
# Create dummy signals
prices = pd.Series([100, 101, 102], name='Price')
signals = pd.DataFrame(index=prices.index)
signals['price'] = prices
signals['positions'] = 0.0

simulate_trading(signals, quiet=True)

# Verify print was not called
mock_print.assert_not_called()

@patch('builtins.print')
def test_simulate_trading_verbose(self, mock_print):
# Create dummy signals
prices = pd.Series([100, 101, 102], name='Price')
signals = pd.DataFrame(index=prices.index)
signals['price'] = prices
signals['positions'] = 0.0

simulate_trading(signals, quiet=False)

# Verify print was called (at least for the header)
self.assertTrue(mock_print.called)

def test_main_runs(self):
# Run main with small number of days to ensure no errors
# We can suppress output with quiet=True
try:
main(days=5, initial_price=100, volatility=0.01, initial_cash=1000, quiet=True, no_color=True)
except Exception as e:
self.fail(f"main() raised {e} unexpectedly!")

if __name__ == '__main__':
unittest.main()