feat(deployment): Add deployment script and signal tracking system

배포 및 실전 운영 준비:

1. 배포 스크립트 (deploy.sh)
   - SSH 기반 자동 배포
   - 원격 백업 생성
   - nginx 자동 재시작
   - 헬스 체크

2. Live Outcome Ledger (live_outcome_ledger.gs)
   - addSignal_(): 신호 기록
   - updatePriceT5_(): T+5 가격 입력
   - updatePriceT20_(): T+20 가격 + outcome 자동 계산
   - calculateStats_(): 통계 계산 (win_rate, avg_margin)
   - checkCalibrationReady_(): CALIBRATED 전환 조건 확인
   - calibrateIfReady_(): 자동 전환 (30개 신호 + 60% 승률)

3. 일일 추적 가이드 (DAILY_SIGNAL_TRACKING.md)
   - 신호 발생 시 → T+5 → T+20 프로세스
   - 주간 리뷰 체크리스트
   - 마일스톤 일정 (6주)
   - CALIBRATED 전환 조건
   - honest_proof_score 개선 경로

배포 준비:
  - publish 폴더: 24MB (172개 파일)
  - appsettings.json: PostgreSQL 연결 설정됨
  - MudBlazor UI: 반응형 대시보드
  - GAS 함수: 7개 (P3~P6)

실전 운영:
  - 신호 수집 기간: 2026-06-25 ~ 2026-08-10 (6주)
  - 목표: 30개 신호 + win_rate >= 60%
  - 최종 목표: honest_proof_score 95.0 달성

Co-Authored-By: Claude Haiku 4.5 <noreply@anthropic.com>
This commit is contained in:
2026-06-25 17:57:50 +09:00
parent 0a51702a9a
commit 2c49f083d0
3 changed files with 761 additions and 0 deletions
+123
View File
@@ -0,0 +1,123 @@
#!/bin/bash
# Quant Engine Web Service Deployment Script
# 목표: publish 폴더를 웹 서버에 배포
set -e
# 설정
SOURCE_DIR="src/dotnet/QuantEngine.Web/publish"
DEPLOY_USER="kjh2064"
DEPLOY_HOST="178.104.200.7"
DEPLOY_PATH="/var/www/quant"
SSH_KEY="${HOME}/.ssh/id_ed25519"
echo "🚀 Quant Engine 웹 서비스 배포 시작"
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo "소스: $SOURCE_DIR"
echo "대상: $DEPLOY_USER@$DEPLOY_HOST:$DEPLOY_PATH"
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
# 1. 배포 폴더 생성/준비
echo ""
echo "📦 Step 1: 배포 폴더 준비..."
if [ ! -d "$SOURCE_DIR" ]; then
echo "❌ 오류: publish 폴더 없음. 먼저 'dotnet publish -c Release'를 실행하세요"
exit 1
fi
echo "✓ publish 폴더 크기: $(du -sh $SOURCE_DIR | cut -f1)"
echo "✓ 파일 수: $(find $SOURCE_DIR -type f | wc -l)"
# 2. SSH 연결 확인
echo ""
echo "🔐 Step 2: SSH 연결 확인..."
if [ ! -f "$SSH_KEY" ]; then
echo "❌ SSH 키 없음: $SSH_KEY"
exit 1
fi
ssh -i "$SSH_KEY" -o ConnectTimeout=10 "$DEPLOY_USER@$DEPLOY_HOST" "echo '✓ SSH 연결 성공'" || {
echo "❌ SSH 연결 실패"
exit 1
}
# 3. 원격 백업
echo ""
echo "💾 Step 3: 원격 백업 생성..."
BACKUP_DIR="/var/www/quant_backup_$(date +%Y%m%d_%H%M%S)"
ssh -i "$SSH_KEY" "$DEPLOY_USER@$DEPLOY_HOST" \
"sudo mkdir -p $DEPLOY_PATH && \
if [ -d $DEPLOY_PATH/publish ]; then \
sudo cp -r $DEPLOY_PATH/publish $BACKUP_DIR; \
echo '✓ 백업 생성: $BACKUP_DIR'; \
else \
echo '✓ 기존 배포 없음'; \
fi"
# 4. 배포
echo ""
echo "📤 Step 4: 파일 전송 중... (이 작업은 시간이 걸릴 수 있습니다)"
rsync -av -e "ssh -i $SSH_KEY" \
--delete \
"$SOURCE_DIR/" \
"$DEPLOY_USER@$DEPLOY_HOST:$DEPLOY_PATH/publish/" \
|| {
echo "❌ 배포 실패"
exit 1
}
echo "✓ 파일 전송 완료"
# 5. 권한 설정
echo ""
echo "🔧 Step 5: 원격 권한 설정..."
ssh -i "$SSH_KEY" "$DEPLOY_USER@$DEPLOY_HOST" \
"sudo chown -R www-data:www-data $DEPLOY_PATH/publish && \
sudo chmod -R 755 $DEPLOY_PATH/publish && \
echo '✓ 권한 설정 완료'"
# 6. 웹 서버 재시작
echo ""
echo "🔄 Step 6: 웹 서버 재시작 중..."
ssh -i "$SSH_KEY" "$DEPLOY_USER@$DEPLOY_HOST" \
"sudo systemctl restart nginx && \
sleep 2 && \
sudo systemctl status nginx | grep Active && \
echo '✓ nginx 재시작 완료'" \
|| {
echo "⚠️ nginx 재시작 실패 (수동으로 확인 필요)"
}
# 7. 배포 확인
echo ""
echo "🧪 Step 7: 배포 확인..."
sleep 2
HEALTH_URL="http://178.104.200.7/quant/"
HTTP_CODE=$(curl -s -o /dev/null -w "%{http_code}" "$HEALTH_URL" || echo "000")
if [ "$HTTP_CODE" = "200" ]; then
echo "✅ 배포 성공! URL: $HEALTH_URL"
elif [ "$HTTP_CODE" = "301" ] || [ "$HTTP_CODE" = "302" ]; then
echo "✓ 배포 완료 (리다이렉트: $HTTP_CODE)"
else
echo "⚠️ HTTP 상태: $HTTP_CODE (nginx 설정 확인 필요)"
fi
# 8. 최종 보고
echo ""
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo "✅ 배포 완료!"
echo "━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━"
echo ""
echo "📋 배포 정보:"
echo " 웹사이트: http://178.104.200.7/quant/"
echo " 배포 경로: $DEPLOY_PATH/publish"
echo " 백업 위치: $BACKUP_DIR (필요시)"
echo ""
echo "🔍 로그 확인:"
echo " ssh $DEPLOY_USER@$DEPLOY_HOST"
echo " sudo tail -f /var/log/nginx/error.log"
echo " sudo tail -f /var/log/nginx/access.log"
echo ""
exit 0
+274
View File
@@ -0,0 +1,274 @@
# 📊 Daily Signal Tracking Guide
**목표**: 30개 거래신호 수집 → CALIBRATED 전환 → honest_proof_score 95 달성
**기간**: 2026-06-25 ~ 2026-08-10 (약 6주)
---
## 📋 매일 해야 할 일
### 1️⃣ 신호 발생 시 (거래 진입 시점)
```python
# Python 또는 GAS 콘솔에서 실행
signal = {
"date": "2026-06-25",
"ticker": "000660", # SK하이닉스 등
"signal_type": "BUY", # BUY 또는 SELL
"signal_score": 78, # 0-100
"entry_price": 50000, # KRW
"entry_quantity": 10, # 주
"entry_time": "10:30", # HH:MM
"style": "SWING", # SCALP|SWING|MOMENTUM|POSITION
"routing_confidence": 82, # buildRoutePacket_ 결과
"notes": "MA20 돌파 + 스마트머니 매수"
}
# GAS: addSignal_(signal)
# 또는 스프레드시트에 직접 입력
```
**✅ 체크리스트:**
- [ ] signal_id 자동 생성됨 (YYYYMMDD_HHMM 형식)
- [ ] validation_status = "UNVALIDATED"
- [ ] 스프레드시트 행 추가됨
---
### 2️⃣ T+5 (5거래일 후)
```
거래일 기준:
- 월요일 진입 → 다음주 월요일이 T+5
- 금요일 진입 → 그다음주 금요일이 T+5
```
**해야 할 일:**
1. T+5일의 종가 조회
2. `updatePriceT5_(signalId, priceT5)` 실행
3. 또는 스프레드시트 "price_t5" 열에 직접 입력
**예시:**
```
signal_id: 20260625_1030
진입가: 50,000
T+5 종가: 51,000
```
---
### 3️⃣ T+20 (20거래일 후) ⭐ 가장 중요
```
T+5 이후 추가 15거래일 경과
```
**해야 할 일:**
1. T+20 종가 조회
2. `updatePriceT20_(signalId, priceT20)` 실행
3. **자동으로 계산됨:**
- `return_pct_t20` = (priceT20 - entryPrice) / entryPrice * 100
- `outcome` = WIN / LOSS / BREAKEVEN
- `win_margin` = |return_pct_t20|
- `validation_status` = PROVISIONAL (자동으로 UNVALIDATED → PROVISIONAL 전환)
**판정 기준:**
```
return_pct_t20 > 2% → WIN
-2% ≤ ret_pct ≤ 2% → BREAKEVEN (통계 제외)
return_pct_t20 < -2% → LOSS
```
**예시:**
```
signal_id: 20260625_1030
진입가: 50,000
T+20 종가: 51,050
수익률: (51,050-50,000)/50,000 * 100 = 2.1%
outcome: WIN ✅
win_margin: 2.1
validation_status: PROVISIONAL
```
---
## 📈 주간 리뷰 (매주 금요일)
### 확인 사항
```javascript
// GAS 콘솔에서 실행
stats = calculateStats_();
Logger.log(JSON.stringify(stats, null, 2));
```
**출력 예시:**
```json
{
"total": 8,
"completed": 4,
"win_count": 3,
"loss_count": 1,
"breakeven_count": 0,
"win_rate": "75.00",
"avg_win_margin": "2.45",
"calibrated_progress": "4/30"
}
```
### 분석
-**win_rate >= 60%?** → YES면 순조로운 진행
- 📊 **avg_win_margin** → 평균 수익률 확인
- 🎯 **calibrated_progress** → 남은 신호 수 (30 - 완료)
### 보고
```markdown
## 주간 리포트 (Week 1)
| 항목 | 값 |
|------|-----|
| 누적 신호 | 8개 |
| 완료됨 | 4개 |
| 승률 | 75% |
| 평균 수익 | 2.45% |
| 진행률 | 4/30 |
| 예상 완료 | 2026-07-20 |
```
---
## 🎯 마일스톤
### Week 1-2 (2026-06-25 ~ 2026-07-08)
- **목표**: 6-8개 신호
- **누적**: 6-8개
- **예상 승률**: 50-70%
### Week 3-4 (2026-07-09 ~ 2026-07-22)
- **목표**: 추가 8-10개
- **누적**: 14-18개
- **T+20 데이터 수집 시작** (첫 신호들 마감)
### Week 5-6 (2026-07-23 ~ 2026-08-05)
- **목표**: 추가 8-10개
- **누적**: 22-28개
- **승률 검증** 시작
### Week 7 (2026-08-06 ~ 2026-08-10)
- **목표**: 최종 2-8개
- **누적**: 30개 완료
- **CALIBRATED 전환 확인**
---
## 🚀 CALIBRATED 전환
### 자동 확인
```javascript
// 매일 또는 주간 실행
check = checkCalibrationReady_();
Logger.log(JSON.stringify(check, null, 2));
```
### 조건
```
✅ sample_count >= 30
✅ avg_win_rate >= 60%
```
### 전환 프로세스
```javascript
// 조건 충족 시 실행
calibrateIfReady_();
// 결과
// → 모든 PROVISIONAL → CALIBRATED
// → honest_proof_score +15점 (86.57 → 101.57... 실제로는 cap 95)
// → 알고리즘 locked 배포
```
---
## 📊 honest_proof_score 개선 경로
```
현재: 56.57
Phase 1 (P0): +10점
→ 66.57
Phase 2 (30건 샘플): +20점
→ 86.57
Phase 3 (P3~P6 운영): +8점
→ 94.57 ≈ 95 목표 달성 ✅
```
---
## ⚠️ 주의사항
### 신호 품질
- **거짓 신호 추가 금지** (spec 위반)
- **뒷북 신호 제외** (P5 Alpha Lead 미충족)
- **배분 위험 신호 차단** (P5 Distribution Risk Gate)
### 데이터 정확성
- **T+20 가격**: KIS/OpenAPI/Yahoo Finance에서 정확하게 수집
- **수익률 계산**: 수수료·세금 제외 (순가격 기준)
- **시간대**: 모든 시간대는 KRW/KST 기준
### 매뉴얼 점검
- 주당 1회 통계 검증
- 월당 1회 샘플 품질 감사
- 승률 급락 시 즉시 신호 정책 재검토
---
## 📝 템플릿
### 신호 기록 양식
```
신호 ID: [자동 생성]
종목: SK하이닉스 (000660)
진입가: 50,000원
진입 수량: 10주
진입 시간: 10:30
신호 강도: 78/100
라우팅 신뢰도: 82/100 (buildRoutePacket_)
스타일: SWING
이유: 5일선 돌파 + 스마트머니 순매수 + 기관 매수
```
### T+20 기록
```
T+20 종가: 51,050원
수익률: +2.1%
판정: WIN
마진: 2.1%
메모: 목표가 도달, 손절 전 청산
```
---
## 🔗 관련 문서
- `spec/realtime/live_outcome_ledger_plan.yaml` — 마스터 계획
- `src/google_apps_script/live_outcome_ledger.gs` — GAS 코드
- `V9_HARDENING_IMPLEMENTATION_ROADMAP.md` — 전체 로드맵
---
**마지막 업데이트**: 2026-06-25
**다음 리뷰**: 2026-07-04 (금요일)
@@ -0,0 +1,364 @@
/**
* Quant Engine v9 — Live Outcome Ledger
* 실전 거래신호 추적 & CALIBRATED 전환 시스템
* 생성: 2026-06-25
*/
// ────────────────────────────────────────────────────────────────────────────
// 설정
// ────────────────────────────────────────────────────────────────────────────
const LEDGER_SHEET_ID = "1N_A..."; // TODO: 스프레드시트 ID 입력
const LEDGER_SHEET_NAME = "live_outcome_ledger";
const LEDGER_HEADERS = [
"signal_id",
"date",
"ticker",
"signal_type",
"signal_score",
"entry_price",
"entry_quantity",
"entry_time",
"style",
"routing_confidence",
"price_t5",
"price_t10",
"price_t20",
"return_pct_t20",
"outcome",
"win_margin",
"validation_status",
"notes",
"last_updated"
];
// ────────────────────────────────────────────────────────────────────────────
// 신호 기록
// ────────────────────────────────────────────────────────────────────────────
/**
* addSignal_
* 새로운 거래 신호를 레저에 추가
*
* @param {Object} signal - 신호 데이터
* @return {string} signal_id
*/
function addSignal_(signal) {
if (!signal.ticker || !signal.signal_type || !signal.entry_price) {
throw new Error("필수 필드 누락: ticker, signal_type, entry_price");
}
const signalId = generateSignalId_(signal.date || new Date());
const ss = SpreadsheetApp.openById(LEDGER_SHEET_ID);
const sheet = ss.getSheetByName(LEDGER_SHEET_NAME);
const row = [
signalId, // signal_id
signal.date || new Date(), // date
signal.ticker, // ticker
signal.signal_type, // signal_type (BUY|SELL)
signal.signal_score || 0, // signal_score
signal.entry_price, // entry_price
signal.entry_quantity || 0, // entry_quantity
signal.entry_time || "00:00", // entry_time
signal.style || "UNCLASSIFIED", // style
signal.routing_confidence || 0, // routing_confidence
null, // price_t5 (T+5에 입력)
null, // price_t10 (T+10에 입력)
null, // price_t20 (T+20에 입력)
null, // return_pct_t20 (자동 계산)
"PENDING", // outcome
null, // win_margin
"UNVALIDATED", // validation_status
signal.notes || "", // notes
new Date().toISOString() // last_updated
];
sheet.appendRow(row);
Logger.log(`✓ 신호 기록: ${signalId} (${signal.ticker})`);
return signalId;
}
/**
* generateSignalId_
* 고유한 signal_id 생성 (YYYYMMDD_HHMM_TICKER 형식)
*
* @param {Date} date
* @return {string} signal_id
*/
function generateSignalId_(date) {
const yyyy = date.getFullYear();
const mm = String(date.getMonth() + 1).padStart(2, "0");
const dd = String(date.getDate()).padStart(2, "0");
const hh = String(date.getHours()).padStart(2, "0");
const min = String(date.getMinutes()).padStart(2, "0");
return `${yyyy}${mm}${dd}_${hh}${min}`;
}
// ────────────────────────────────────────────────────────────────────────────
// T+5, T+10, T+20 가격 입력
// ────────────────────────────────────────────────────────────────────────────
/**
* updatePriceT5_
* T+5 가격 입력 (자동 또는 수동)
*
* @param {string} signalId
* @param {number} priceT5
*/
function updatePriceT5_(signalId, priceT5) {
const ss = SpreadsheetApp.openById(LEDGER_SHEET_ID);
const sheet = ss.getSheetByName(LEDGER_SHEET_NAME);
const rowIndex = findSignalRow_(sheet, signalId);
if (rowIndex === -1) throw new Error(`신호를 찾을 수 없음: ${signalId}`);
sheet.getRange(rowIndex, 11).setValue(priceT5); // price_t5 = 11번째 열
sheet.getRange(rowIndex, 19).setValue(new Date().toISOString()); // last_updated
Logger.log(`✓ T+5 가격 입력: ${signalId} = ${priceT5}`);
}
/**
* updatePriceT20_
* T+20 가격 입력 + outcome 자동 계산
*
* @param {string} signalId
* @param {number} priceT20
*/
function updatePriceT20_(signalId, priceT20) {
const ss = SpreadsheetApp.openById(LEDGER_SHEET_ID);
const sheet = ss.getSheetByName(LEDGER_SHEET_NAME);
const rowIndex = findSignalRow_(sheet, signalId);
if (rowIndex === -1) throw new Error(`신호를 찾을 수 없음: ${signalId}`);
const range = sheet.getRange(rowIndex, 1, 1, LEDGER_HEADERS.length);
const values = range.getValues()[0];
// entry_price (인덱스 5) vs priceT20
const entryPrice = values[5];
const returnPctT20 = ((priceT20 - entryPrice) / entryPrice * 100).toFixed(2);
// outcome 판정: WIN (-2% < ret < 2% = BREAKEVEN), LOSS, WIN
let outcome = "BREAKEVEN";
let winMargin = Math.abs(returnPctT20);
if (returnPctT20 > 2) {
outcome = "WIN";
} else if (returnPctT20 < -2) {
outcome = "LOSS";
winMargin = returnPctT20;
}
// 업데이트
sheet.getRange(rowIndex, 13).setValue(priceT20); // price_t20
sheet.getRange(rowIndex, 14).setValue(returnPctT20); // return_pct_t20
sheet.getRange(rowIndex, 15).setValue(outcome); // outcome
sheet.getRange(rowIndex, 16).setValue(winMargin); // win_margin
sheet.getRange(rowIndex, 17).setValue("PROVISIONAL"); // validation_status
sheet.getRange(rowIndex, 19).setValue(new Date().toISOString()); // last_updated
Logger.log(`✓ T+20 업데이트: ${signalId}, 수익률=${returnPctT20}%, outcome=${outcome}`);
}
// ────────────────────────────────────────────────────────────────────────────
// 통계 & 검증
// ────────────────────────────────────────────────────────────────────────────
/**
* calculateStats_
* 현재까지의 통계 계산 (win_rate, avg_win_margin, etc.)
*
* @return {Object} { total, completed, win_count, loss_count, win_rate, avg_margin }
*/
function calculateStats_() {
const ss = SpreadsheetApp.openById(LEDGER_SHEET_ID);
const sheet = ss.getSheetByName(LEDGER_SHEET_NAME);
const data = sheet.getDataRange().getValues();
let totalSignals = 0;
let completedSignals = 0;
let winCount = 0;
let lossCount = 0;
let totalWinMargin = 0;
// 헤더 제외 (인덱스 1부터)
for (let i = 1; i < data.length; i++) {
const row = data[i];
const validationStatus = row[16]; // validation_status
const outcome = row[14]; // outcome
const winMargin = row[15]; // win_margin
totalSignals++;
if (validationStatus === "PROVISIONAL" || validationStatus === "CALIBRATED") {
completedSignals++;
if (outcome === "WIN") {
winCount++;
totalWinMargin += Math.abs(winMargin);
} else if (outcome === "LOSS") {
lossCount++;
}
}
}
const winRate = completedSignals > 0 ? (winCount / completedSignals * 100) : 0;
const avgMargin = completedSignals > 0 ? (totalWinMargin / winCount || 0) : 0;
return {
total: totalSignals,
completed: completedSignals,
win_count: winCount,
loss_count: lossCount,
breakeven_count: completedSignals - winCount - lossCount,
win_rate: winRate.toFixed(2),
avg_win_margin: avgMargin.toFixed(2),
calibrated_progress: `${completedSignals}/30`
};
}
/**
* checkCalibrationReady_
* CALIBRATED 전환 가능 여부 확인
*
* @return {Object} { ready, reason, stats }
*/
function checkCalibrationReady_() {
const stats = calculateStats_();
const minSamples = 30;
const minWinRate = 60;
const ready = (stats.completed >= minSamples && parseFloat(stats.win_rate) >= minWinRate);
const reason = !ready ?
`샘플: ${stats.completed}/${minSamples}, 승률: ${stats.win_rate}%/${minWinRate}%` :
"✅ CALIBRATED 전환 조건 충족!";
return {
ready: ready,
reason: reason,
stats: stats
};
}
/**
* calibrateIfReady_
* CALIBRATED 전환 (자동 또는 수동)
*/
function calibrateIfReady_() {
const check = checkCalibrationReady_();
if (!check.ready) {
Logger.log(`⏳ CALIBRATED 전환 대기: ${check.reason}`);
return;
}
// 모든 PROVISIONAL을 CALIBRATED로 전환
const ss = SpreadsheetApp.openById(LEDGER_SHEET_ID);
const sheet = ss.getSheetByName(LEDGER_SHEET_NAME);
const data = sheet.getDataRange().getValues();
let calibratedCount = 0;
for (let i = 1; i < data.length; i++) {
if (data[i][16] === "PROVISIONAL") {
sheet.getRange(i + 1, 17).setValue("CALIBRATED");
calibratedCount++;
}
}
Logger.log(`✅ CALIBRATED 전환 완료: ${calibratedCount}개 신호`);
Logger.log(`📊 최종 통계: ${JSON.stringify(check.stats, null, 2)}`);
}
// ────────────────────────────────────────────────────────────────────────────
// 유틸리티
// ────────────────────────────────────────────────────────────────────────────
/**
* findSignalRow_
* signal_id로 행 번호 찾기
*
* @param {Sheet} sheet
* @param {string} signalId
* @return {number} 행 번호 (없으면 -1)
*/
function findSignalRow_(sheet, signalId) {
const data = sheet.getDataRange().getValues();
for (let i = 1; i < data.length; i++) {
if (data[i][0] === signalId) {
return i + 1; // 1-indexed
}
}
return -1;
}
/**
* initializeLedger_
* 레저 시트 초기화 (첫 실행 시만)
*/
function initializeLedger_() {
const ss = SpreadsheetApp.openById(LEDGER_SHEET_ID);
let sheet = ss.getSheetByName(LEDGER_SHEET_NAME);
if (!sheet) {
sheet = ss.insertSheet(LEDGER_SHEET_NAME);
}
// 헤더 설정
sheet.getRange(1, 1, 1, LEDGER_HEADERS.length).setValues([LEDGER_HEADERS]);
// 포맷팅
sheet.getRange(1, 1, 1, LEDGER_HEADERS.length)
.setBackground("#1f77b4")
.setFontColor("white")
.setFontWeight("bold");
Logger.log(`✓ 레저 시트 초기화 완료: ${LEDGER_SHEET_NAME}`);
}
/**
* 테스트 함수
*/
function testLiveOutcomeLedger() {
Logger.log("=== Live Outcome Ledger 테스트 ===");
// 테스트 신호 추가
const testSignal = {
date: new Date(),
ticker: "000660",
signal_type: "BUY",
signal_score: 75,
entry_price: 100000,
entry_quantity: 10,
style: "SWING",
routing_confidence: 80,
notes: "테스트 신호"
};
try {
// 초기화
initializeLedger_();
// 신호 추가
// const signalId = addSignal_(testSignal);
// Logger.log(`신호 ID: ${signalId}`);
// 통계 확인
const stats = calculateStats_();
Logger.log(`통계: ${JSON.stringify(stats, null, 2)}`);
// CALIBRATED 준비 여부
const check = checkCalibrationReady_();
Logger.log(`CALIBRATED 준비: ${JSON.stringify(check, null, 2)}`);
} catch (error) {
Logger.log(`❌ 오류: ${error.message}`);
}
}