deepcoin 패키지를 bithumb으로 rename하고, 3단계 live 운영·사이징 튜닝·텔레그램 알림을 통합한다. Co-authored-by: Cursor <cursoragent@cursor.com>
113 lines
3.4 KiB
Python
113 lines
3.4 KiB
Python
#!/usr/bin/env python3
|
|
"""3단계: paper/live 운영 1회 tick (신호 확인·체결)."""
|
|
|
|
from __future__ import annotations
|
|
|
|
import argparse
|
|
import logging
|
|
import sys
|
|
import time
|
|
from pathlib import Path
|
|
|
|
ROOT = Path(__file__).resolve().parents[1]
|
|
SRC = ROOT / "src"
|
|
if str(SRC) not in sys.path:
|
|
sys.path.insert(0, str(SRC))
|
|
|
|
from bithumb.config import load_settings
|
|
from bithumb.operations.runner import OperationsRunner
|
|
|
|
|
|
def _configure_logging(verbose: bool) -> None:
|
|
level = logging.DEBUG if verbose else logging.INFO
|
|
logging.basicConfig(
|
|
level=level,
|
|
format="%(asctime)s [%(levelname)s] %(message)s",
|
|
datefmt="%Y-%m-%d %H:%M:%S",
|
|
)
|
|
|
|
|
|
def main() -> int:
|
|
"""CLI 진입점."""
|
|
parser = argparse.ArgumentParser(description="3단계: Bithumb 운영 tick")
|
|
parser.add_argument(
|
|
"--mode",
|
|
choices=("paper", "live"),
|
|
default=None,
|
|
help="OPS_MODE 덮어쓰기 (기본 .env)",
|
|
)
|
|
parser.add_argument(
|
|
"--no-sync",
|
|
action="store_true",
|
|
help="캔들 증분 동기화 생략",
|
|
)
|
|
parser.add_argument(
|
|
"--loop",
|
|
type=int,
|
|
default=0,
|
|
metavar="SEC",
|
|
help="N초마다 반복 실행 (0=1회)",
|
|
)
|
|
parser.add_argument("-v", "--verbose", action="store_true")
|
|
args = parser.parse_args()
|
|
_configure_logging(args.verbose)
|
|
|
|
if args.mode:
|
|
import os
|
|
os.environ["OPS_MODE"] = args.mode
|
|
settings = load_settings()
|
|
|
|
if settings.ops_mode == "live":
|
|
if not settings.bithumb_access_key or not settings.bithumb_secret_key:
|
|
print("live 모드에는 BITHUMB_ACCESS_KEY / BITHUMB_SECRET_KEY 가 필요합니다.", file=sys.stderr)
|
|
return 1
|
|
print("경고: live 모드 — 실제 주문이 발생할 수 있습니다.")
|
|
|
|
runner = OperationsRunner(settings)
|
|
sync = not args.no_sync
|
|
|
|
while True:
|
|
try:
|
|
report = runner.tick(sync_candles=sync)
|
|
except Exception as exc:
|
|
logging.exception("운영 tick 예외 (복구 후 계속)")
|
|
if runner.telegram.is_active:
|
|
runner.telegram.notify_ops_error(
|
|
mode=settings.ops_mode,
|
|
symbol=settings.symbol,
|
|
technique_id=settings.ops_technique_id,
|
|
stage="main_loop",
|
|
error=str(exc),
|
|
)
|
|
if args.loop <= 0:
|
|
break
|
|
time.sleep(args.loop)
|
|
continue
|
|
|
|
port = report.get("portfolio") or {}
|
|
print("\n=== 3단계 운영 tick ===")
|
|
if report.get("error"):
|
|
print(f"오류: [{report.get('error_stage')}] {report.get('error_message')}")
|
|
print(f"모드: {report.get('mode', settings.ops_mode)}")
|
|
print(
|
|
f"최신 봉 후보: {report.get('latest_bar_candidates', 0)} · "
|
|
f"필터 통과: {report.get('filtered_signals', 0)} · "
|
|
f"처리 bar: {report.get('pending_bars', [])}"
|
|
)
|
|
print(f"이번 체결: {len(report.get('executions', []))}건")
|
|
print(
|
|
f"포트폴리오: 현금 {port.get('cash_krw', 0):,.0f}원 · "
|
|
f"코인 {port.get('coin_qty', 0):.8f} {settings.symbol}"
|
|
)
|
|
print(f"리포트: {settings.ops_report_json}")
|
|
|
|
if args.loop <= 0:
|
|
break
|
|
time.sleep(args.loop)
|
|
|
|
return 0
|
|
|
|
|
|
if __name__ == "__main__":
|
|
raise SystemExit(main())
|